Skip to content

Commit 4f68c04

Browse files
committed
Add a test suite for _pyrepl.layout
1 parent 999c94e commit 4f68c04

File tree

1 file changed

+310
-0
lines changed

1 file changed

+310
-0
lines changed
Lines changed: 310 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,310 @@
1+
from unittest import TestCase
2+
from _pyrepl.content import (
3+
ContentFragment,
4+
ContentLine,
5+
PromptContent,
6+
SourceLine,
7+
)
8+
from _pyrepl.layout import (
9+
LayoutMap,
10+
LayoutRow,
11+
WrappedRow,
12+
layout_content_lines,
13+
)
14+
15+
16+
def _source(text, lineno=0, start_offset=0, has_newline=True, cursor_index=None):
17+
return SourceLine(lineno, text, start_offset, has_newline, cursor_index)
18+
19+
20+
def _prompt(text=">>> ", width=4):
21+
return PromptContent((), text, width)
22+
23+
24+
def _body_from_text(text):
25+
return tuple(ContentFragment(ch, 1) for ch in text)
26+
27+
28+
def _content_line(text, prompt=None, has_newline=True, start_offset=0):
29+
if prompt is None:
30+
prompt = _prompt()
31+
body = _body_from_text(text)
32+
source = _source(text, start_offset=start_offset, has_newline=has_newline)
33+
return ContentLine(source, prompt, body)
34+
35+
36+
class TestLayoutRow(TestCase):
37+
def test_width_basic(self):
38+
row = LayoutRow(4, (1, 1, 1))
39+
self.assertEqual(row.width, 7)
40+
41+
def test_width_with_suffix(self):
42+
row = LayoutRow(4, (1, 1), suffix_width=1)
43+
self.assertEqual(row.width, 7)
44+
45+
def test_screeninfo_without_suffix(self):
46+
row = LayoutRow(4, (1, 1, 1))
47+
prompt_w, widths = row.screeninfo
48+
self.assertEqual(prompt_w, 4)
49+
self.assertEqual(widths, [1, 1, 1])
50+
51+
def test_screeninfo_with_suffix(self):
52+
row = LayoutRow(4, (1, 1), suffix_width=1)
53+
prompt_w, widths = row.screeninfo
54+
self.assertEqual(prompt_w, 4)
55+
self.assertEqual(widths, [1, 1, 1])
56+
57+
58+
class TestLayoutMap(TestCase):
59+
def test_empty(self):
60+
lm = LayoutMap.empty()
61+
self.assertEqual(len(lm.rows), 1)
62+
self.assertEqual(lm.max_row(), 0)
63+
self.assertEqual(lm.max_column(0), 0)
64+
65+
def test_screeninfo(self):
66+
lm = LayoutMap((
67+
LayoutRow(4, (1, 1, 1)),
68+
LayoutRow(0, (1, 1)),
69+
))
70+
info = lm.screeninfo
71+
self.assertEqual(len(info), 2)
72+
self.assertEqual(info[0], (4, [1, 1, 1]))
73+
self.assertEqual(info[1], (0, [1, 1]))
74+
75+
def test_max_column(self):
76+
lm = LayoutMap((LayoutRow(4, (1, 1, 1)),))
77+
self.assertEqual(lm.max_column(0), 7)
78+
79+
def test_max_row(self):
80+
lm = LayoutMap((LayoutRow(0, ()), LayoutRow(0, ())))
81+
self.assertEqual(lm.max_row(), 1)
82+
83+
def test_pos_to_xy_empty_rows(self):
84+
lm = LayoutMap(())
85+
self.assertEqual(lm.pos_to_xy(0), (0, 0))
86+
87+
def test_pos_to_xy_single_row(self):
88+
lm = LayoutMap((LayoutRow(4, (1, 1, 1), buffer_advance=3),))
89+
self.assertEqual(lm.pos_to_xy(0), (4, 0))
90+
self.assertEqual(lm.pos_to_xy(1), (5, 0))
91+
self.assertEqual(lm.pos_to_xy(2), (6, 0))
92+
self.assertEqual(lm.pos_to_xy(3), (7, 0))
93+
94+
def test_pos_to_xy_multi_row(self):
95+
lm = LayoutMap((
96+
LayoutRow(4, (1, 1, 1), buffer_advance=4),
97+
LayoutRow(4, (1, 1), buffer_advance=2),
98+
))
99+
# First row: pos 0-3
100+
self.assertEqual(lm.pos_to_xy(0), (4, 0))
101+
self.assertEqual(lm.pos_to_xy(3), (7, 0))
102+
# Second row: pos 4-5
103+
self.assertEqual(lm.pos_to_xy(4), (4, 1))
104+
self.assertEqual(lm.pos_to_xy(5), (5, 1))
105+
106+
def test_pos_to_xy_past_end_clamps(self):
107+
lm = LayoutMap((LayoutRow(4, (1, 1), buffer_advance=2),))
108+
self.assertEqual(lm.pos_to_xy(99), (6, 0))
109+
110+
def test_pos_to_xy_skips_prompt_only_leading_rows(self):
111+
lm = LayoutMap((
112+
LayoutRow(0, (), buffer_advance=0), # leading prompt-only row
113+
LayoutRow(4, (1, 1), buffer_advance=3),
114+
))
115+
# pos 0 should skip the leading row and land on the real row
116+
self.assertEqual(lm.pos_to_xy(0), (4, 1))
117+
118+
def test_pos_to_xy_with_suffix(self):
119+
lm = LayoutMap((
120+
LayoutRow(4, (1, 1), suffix_width=1, buffer_advance=2),
121+
LayoutRow(0, (1,), buffer_advance=2),
122+
))
123+
# pos=2 fits within first row's char_widths (len=2), cursor at end
124+
self.assertEqual(lm.pos_to_xy(2), (6, 0))
125+
# pos=3 exceeds first row, lands on second row
126+
self.assertEqual(lm.pos_to_xy(3), (1, 1))
127+
128+
def test_xy_to_pos_empty_rows(self):
129+
lm = LayoutMap(())
130+
self.assertEqual(lm.xy_to_pos(0, 0), 0)
131+
132+
def test_xy_to_pos_single_row(self):
133+
lm = LayoutMap((LayoutRow(4, (1, 1, 1), buffer_advance=3),))
134+
self.assertEqual(lm.xy_to_pos(4, 0), 0)
135+
self.assertEqual(lm.xy_to_pos(5, 0), 1)
136+
self.assertEqual(lm.xy_to_pos(6, 0), 2)
137+
self.assertEqual(lm.xy_to_pos(7, 0), 3)
138+
139+
def test_xy_to_pos_multi_row(self):
140+
lm = LayoutMap((
141+
LayoutRow(4, (1, 1, 1), buffer_advance=4),
142+
LayoutRow(4, (1, 1), buffer_advance=2),
143+
))
144+
self.assertEqual(lm.xy_to_pos(4, 0), 0)
145+
self.assertEqual(lm.xy_to_pos(4, 1), 4)
146+
self.assertEqual(lm.xy_to_pos(5, 1), 5)
147+
148+
def test_xy_to_pos_before_prompt_returns_zero(self):
149+
lm = LayoutMap((LayoutRow(4, (1, 1), buffer_advance=2),))
150+
self.assertEqual(lm.xy_to_pos(0, 0), 0)
151+
152+
def test_xy_to_pos_with_zero_width_chars(self):
153+
# Simulates combining characters (zero-width) after a base char
154+
lm = LayoutMap((LayoutRow(4, (1, 0, 1), buffer_advance=3),))
155+
# x=5 is past the first char; trailing zero-width combining is included
156+
self.assertEqual(lm.xy_to_pos(5, 0), 2)
157+
158+
def test_xy_to_pos_zero_width_skipped(self):
159+
lm = LayoutMap((LayoutRow(0, (0, 1, 1), buffer_advance=3),))
160+
# x=0: the zero-width char at index 0 is skipped, pos advances
161+
self.assertEqual(lm.xy_to_pos(0, 0), 1)
162+
163+
def test_xy_to_pos_zero_width_before_target(self):
164+
# Zero-width char between two normal chars; target x is past it
165+
lm = LayoutMap((LayoutRow(0, (1, 0, 1), buffer_advance=3),))
166+
# x=2: passes char at x=0 (w=1), skips zero-width at x=1, lands at x=2
167+
self.assertEqual(lm.xy_to_pos(2, 0), 3)
168+
169+
def test_xy_to_pos_trailing_all_zero_width(self):
170+
# All remaining chars from cursor position are zero-width
171+
lm = LayoutMap((LayoutRow(0, (1, 0), buffer_advance=2),))
172+
# x=1: past first char, trailing loop exhausts (all remaining are 0-width)
173+
self.assertEqual(lm.xy_to_pos(1, 0), 2)
174+
175+
176+
class TestLayoutContentLines(TestCase):
177+
def test_zero_width_returns_empty(self):
178+
result = layout_content_lines((), 0, 0)
179+
self.assertEqual(result.wrapped_rows, ())
180+
self.assertEqual(result.layout_map.rows, ())
181+
182+
def test_negative_width_returns_empty(self):
183+
result = layout_content_lines((), -1, 0)
184+
self.assertEqual(result.wrapped_rows, ())
185+
186+
def test_single_short_line(self):
187+
line = _content_line("abc")
188+
result = layout_content_lines((line,), 80, 0)
189+
190+
self.assertEqual(len(result.wrapped_rows), 1)
191+
row = result.wrapped_rows[0]
192+
self.assertEqual(row.prompt_text, ">>> ")
193+
self.assertEqual(row.prompt_width, 4)
194+
self.assertEqual(row.suffix, "")
195+
self.assertEqual(row.buffer_advance, 4) # 3 chars + newline
196+
197+
def test_single_line_no_newline(self):
198+
line = _content_line("ab", has_newline=False)
199+
result = layout_content_lines((line,), 80, 0)
200+
201+
self.assertEqual(len(result.wrapped_rows), 1)
202+
self.assertEqual(result.wrapped_rows[0].buffer_advance, 2)
203+
204+
def test_empty_body(self):
205+
source = _source("", has_newline=True)
206+
prompt = _prompt()
207+
line = ContentLine(source, prompt, ())
208+
result = layout_content_lines((line,), 80, 0)
209+
210+
self.assertEqual(len(result.wrapped_rows), 1)
211+
self.assertEqual(result.wrapped_rows[0].buffer_advance, 1) # just newline
212+
213+
def test_line_wraps(self):
214+
# prompt ">>> " is 4 wide, terminal is 10 wide, so 6 chars fit per row
215+
line = _content_line("abcdefgh") # 8 chars, needs 2 rows
216+
result = layout_content_lines((line,), 10, 0)
217+
218+
self.assertEqual(len(result.wrapped_rows), 2)
219+
first, second = result.wrapped_rows
220+
self.assertEqual(first.prompt_text, ">>> ")
221+
self.assertEqual(first.suffix, "\\")
222+
self.assertEqual(first.suffix_width, 1)
223+
# First row: 10 - 4(prompt) - 1(suffix) = 5 chars
224+
self.assertEqual(first.buffer_advance, 5)
225+
# Second row: continuation with no prompt
226+
self.assertEqual(second.prompt_text, "")
227+
self.assertEqual(second.prompt_width, 0)
228+
self.assertEqual(second.buffer_advance, 4) # remaining 3 + newline
229+
self.assertEqual(second.suffix, "")
230+
231+
def test_wrapping_forces_progress(self):
232+
# When a single character is wider than available space, force 1 char
233+
prompt = _prompt("P", 1)
234+
body = (ContentFragment("W", 1),)
235+
source = _source("W", has_newline=False)
236+
line = ContentLine(source, prompt, body)
237+
# width=2 means prompt(1) + char(1) = 2, which fits (< width would be
238+
# false for width=2 since 1+1 >= 2), so it wraps but forces progress
239+
result = layout_content_lines((line,), 2, 0)
240+
241+
self.assertEqual(len(result.wrapped_rows), 1)
242+
self.assertEqual(result.wrapped_rows[0].buffer_advance, 1)
243+
244+
def test_layout_map_matches_wrapped_rows(self):
245+
line = _content_line("abc")
246+
result = layout_content_lines((line,), 80, 0)
247+
248+
self.assertEqual(len(result.layout_map.rows), len(result.wrapped_rows))
249+
self.assertEqual(result.layout_map.rows[0].prompt_width, 4)
250+
self.assertEqual(result.layout_map.rows[0].char_widths, (1, 1, 1))
251+
252+
def test_line_end_offsets(self):
253+
line1 = _content_line("ab")
254+
line2 = _content_line("cd")
255+
result = layout_content_lines((line1, line2), 80, 0)
256+
257+
self.assertEqual(len(result.line_end_offsets), 2)
258+
# line1: 2 chars + 1 newline = offset 3
259+
self.assertEqual(result.line_end_offsets[0], 3)
260+
# line2: offset 3 + 2 chars + 1 newline = 6
261+
self.assertEqual(result.line_end_offsets[1], 6)
262+
263+
def test_start_offset_shifts_offsets(self):
264+
line = _content_line("ab")
265+
result = layout_content_lines((line,), 80, 10)
266+
267+
self.assertEqual(result.line_end_offsets[0], 13)
268+
269+
def test_multiple_lines(self):
270+
line1 = _content_line("abc")
271+
line2 = _content_line("de")
272+
result = layout_content_lines((line1, line2), 80, 0)
273+
274+
self.assertEqual(len(result.wrapped_rows), 2)
275+
self.assertEqual(result.wrapped_rows[0].buffer_advance, 4) # abc + \n
276+
self.assertEqual(result.wrapped_rows[1].buffer_advance, 3) # de + \n
277+
278+
def test_leading_prompt_lines(self):
279+
leading = (ContentFragment("header", 6),)
280+
prompt = PromptContent(leading, ">>> ", 4)
281+
body = _body_from_text("x")
282+
source = _source("x", has_newline=False)
283+
line = ContentLine(source, prompt, body)
284+
result = layout_content_lines((line,), 80, 0)
285+
286+
# Leading line + body line
287+
self.assertEqual(len(result.wrapped_rows), 2)
288+
# Leading row has the fragment but no prompt
289+
self.assertEqual(result.wrapped_rows[0].fragments, leading)
290+
# Body row has prompt and content
291+
self.assertEqual(result.wrapped_rows[1].prompt_text, ">>> ")
292+
293+
def test_wrapped_line_layout_rows_have_suffix(self):
294+
line = _content_line("abcdefgh")
295+
result = layout_content_lines((line,), 10, 0)
296+
297+
first_layout = result.layout_map.rows[0]
298+
self.assertEqual(first_layout.suffix_width, 1)
299+
second_layout = result.layout_map.rows[1]
300+
self.assertEqual(second_layout.suffix_width, 0)
301+
302+
def test_pos_to_xy_through_layout(self):
303+
line = _content_line("abc")
304+
result = layout_content_lines((line,), 80, 0)
305+
lm = result.layout_map
306+
307+
self.assertEqual(lm.pos_to_xy(0), (4, 0)) # after prompt
308+
self.assertEqual(lm.pos_to_xy(1), (5, 0))
309+
self.assertEqual(lm.pos_to_xy(2), (6, 0))
310+
self.assertEqual(lm.pos_to_xy(3), (7, 0)) # end of line

0 commit comments

Comments
 (0)