git_worktree_manager/tui/
arrow_select.rs1use std::io::{IsTerminal, Write};
5
6pub fn arrow_select(
20 items: &[(String, String)],
21 title: &str,
22 default_index: usize,
23) -> Option<String> {
24 if items.is_empty() {
25 return None;
26 }
27
28 if !std::io::stderr().is_terminal() {
29 return None;
30 }
31
32 let default_index = default_index.min(items.len() - 1);
33
34 #[cfg(unix)]
36 {
37 if let Some(result) = arrow_select_unix(items, title, default_index) {
38 return result;
39 }
40 }
41
42 arrow_select_fallback(items, title, default_index)
44}
45
46#[cfg(unix)]
52fn get_terminal_width() -> usize {
53 console::Term::stderr().size().1 as usize
54}
55
56#[cfg(unix)]
58fn write_stderr(s: &str) {
59 let stderr = std::io::stderr();
60 let mut handle = stderr.lock();
61 let _ = handle.write_all(s.as_bytes());
62 let _ = handle.flush();
63}
64
65#[cfg(any(unix, test))]
67fn visible_len(text: &str) -> usize {
68 let mut len = 0;
69 let bytes = text.as_bytes();
70 let mut i = 0;
71 while i < bytes.len() {
72 if bytes[i] == b'\x1b' {
73 i += 1;
75 while i < bytes.len() && bytes[i] != b'm' {
76 i += 1;
77 }
78 if i < bytes.len() {
79 i += 1; }
81 } else {
82 len += 1;
83 i += 1;
84 }
85 }
86 len
87}
88
89#[cfg(any(unix, test))]
91fn truncate(text: &str, width: usize) -> String {
92 if visible_len(text) <= width {
93 return text.to_string();
94 }
95
96 let mut vis_pos = 0;
97 let mut cut_pos = 0;
98 let bytes = text.as_bytes();
99 let mut i = 0;
100
101 while i < bytes.len() && vis_pos < width.saturating_sub(1) {
102 if bytes[i] == b'\x1b' {
103 i += 1;
105 while i < bytes.len() && bytes[i] != b'm' {
106 i += 1;
107 }
108 if i < bytes.len() {
109 i += 1; }
111 } else {
112 vis_pos += 1;
113 i += 1;
114 }
115 cut_pos = i;
116 }
117
118 let mut result = text[..cut_pos].to_string();
119 result.push_str("\x1b[0m");
120 result
121}
122
123#[cfg(unix)]
129fn render(
130 items: &[(String, String)],
131 title: &str,
132 selected: usize,
133 _total_lines: usize,
134 first_render: bool,
135) {
136 let width = get_terminal_width();
137
138 if !first_render {
139 write_stderr("\x1b[u");
141 }
142
143 write_stderr("\x1b[s");
145
146 let line = format!(" \x1b[1m{title}\x1b[0m");
148 write_stderr(&format!("\x1b[2K{}\r\n", truncate(&line, width)));
149 write_stderr("\x1b[2K\r\n");
151
152 for (i, (label, value)) in items.iter().enumerate() {
153 write_stderr("\x1b[2K"); let line = if i == selected {
155 format!(" \x1b[1;7m > {label} \x1b[0m \x1b[2m{value}\x1b[0m")
156 } else {
157 format!(" {label} \x1b[2m{value}\x1b[0m")
158 };
159 write_stderr(&format!("{}\r\n", truncate(&line, width)));
160 }
161
162 for _ in 0..2 {
164 write_stderr("\x1b[2K\r\n");
165 }
166 write_stderr("\x1b[2A");
168}
169
170#[cfg(unix)]
172fn cleanup(total_lines: usize) {
173 write_stderr("\x1b[u");
175 for _ in 0..total_lines + 2 {
176 write_stderr("\x1b[2K\r\n");
177 }
178 write_stderr("\x1b[u");
179}
180
181#[cfg(unix)]
187#[derive(Debug, PartialEq)]
188enum Key {
189 Up,
190 Down,
191 Enter,
192 Escape,
193 CtrlC,
194 Quit,
195 Number(u8),
196 Unknown,
197}
198
199#[cfg(unix)]
201fn read_key(fd: std::os::unix::io::RawFd) -> Result<Key, std::io::Error> {
202 let mut buf = [0u8; 1];
203 let n = unsafe { libc::read(fd, buf.as_mut_ptr() as *mut libc::c_void, 1) };
204 if n <= 0 {
205 return Err(std::io::Error::new(
206 std::io::ErrorKind::UnexpectedEof,
207 "EOF on stdin",
208 ));
209 }
210
211 match buf[0] {
212 b'\x1b' => {
213 let mut pollfd = libc::pollfd {
215 fd,
216 events: libc::POLLIN,
217 revents: 0,
218 };
219 let ready = unsafe { libc::poll(&mut pollfd as *mut libc::pollfd, 1, 50) };
220 if ready <= 0 {
221 return Ok(Key::Escape);
223 }
224 let mut seq1 = [0u8; 1];
225 let n = unsafe { libc::read(fd, seq1.as_mut_ptr() as *mut libc::c_void, 1) };
226 if n <= 0 {
227 return Ok(Key::Escape);
228 }
229 if seq1[0] == b'[' {
230 let mut seq2 = [0u8; 1];
231 let n = unsafe { libc::read(fd, seq2.as_mut_ptr() as *mut libc::c_void, 1) };
232 if n <= 0 {
233 return Ok(Key::Unknown);
234 }
235 match seq2[0] {
236 b'A' => Ok(Key::Up),
237 b'B' => Ok(Key::Down),
238 _ => Ok(Key::Unknown),
239 }
240 } else {
241 Ok(Key::Unknown)
242 }
243 }
244 b'\r' | b'\n' => Ok(Key::Enter),
245 0x03 => Ok(Key::CtrlC),
246 b'q' => Ok(Key::Quit),
247 c @ b'1'..=b'9' => Ok(Key::Number(c - b'0')),
248 _ => Ok(Key::Unknown),
249 }
250}
251
252#[cfg(unix)]
257fn arrow_select_unix(
258 items: &[(String, String)],
259 title: &str,
260 default_index: usize,
261) -> Option<Option<String>> {
262 use std::os::unix::io::AsRawFd;
263
264 let stdin = std::io::stdin();
265 let fd = stdin.as_raw_fd();
266
267 let mut old_termios: libc::termios = unsafe { std::mem::zeroed() };
269 if unsafe { libc::tcgetattr(fd, &mut old_termios) } != 0 {
270 return None; }
272
273 let mut selected = default_index;
274 let total_lines = items.len() + 2; write_stderr("\x1b[?25l");
278
279 let mut raw = old_termios;
281 raw.c_iflag &= !(libc::IGNBRK
283 | libc::BRKINT
284 | libc::PARMRK
285 | libc::ISTRIP
286 | libc::INLCR
287 | libc::IGNCR
288 | libc::ICRNL
289 | libc::IXON);
290 raw.c_oflag &= !libc::OPOST;
291 raw.c_lflag &= !(libc::ECHO | libc::ECHONL | libc::ICANON | libc::ISIG | libc::IEXTEN);
292 raw.c_cflag &= !(libc::CSIZE | libc::PARENB);
293 raw.c_cflag |= libc::CS8;
294 raw.c_cc[libc::VMIN] = 1;
295 raw.c_cc[libc::VTIME] = 0;
296
297 if unsafe { libc::tcsetattr(fd, libc::TCSAFLUSH, &raw) } != 0 {
298 write_stderr("\x1b[?25h");
299 return None;
300 }
301
302 let result = (|| -> Option<String> {
303 render(items, title, selected, total_lines, true);
304
305 loop {
306 let key = match read_key(fd) {
307 Ok(k) => k,
308 Err(_) => {
309 cleanup(total_lines);
310 return None;
311 }
312 };
313
314 match key {
315 Key::Enter => {
316 cleanup(total_lines);
317 return Some(items[selected].1.clone());
318 }
319 Key::CtrlC | Key::Quit | Key::Escape => {
320 cleanup(total_lines);
321 return None;
322 }
323 Key::Up => {
324 selected = if selected == 0 {
325 items.len() - 1
326 } else {
327 selected - 1
328 };
329 render(items, title, selected, total_lines, false);
330 }
331 Key::Down => {
332 selected = (selected + 1) % items.len();
333 render(items, title, selected, total_lines, false);
334 }
335 Key::Number(n) => {
336 let idx = (n as usize) - 1;
337 if idx < items.len() {
338 cleanup(total_lines);
339 return Some(items[idx].1.clone());
340 }
341 }
342 Key::Unknown => {}
343 }
344 }
345 })();
346
347 unsafe {
349 libc::tcsetattr(fd, libc::TCSADRAIN, &old_termios);
350 }
351 write_stderr("\x1b[?25h");
353
354 Some(result)
355}
356
357fn arrow_select_fallback(
363 items: &[(String, String)],
364 title: &str,
365 default_index: usize,
366) -> Option<String> {
367 let stderr = std::io::stderr();
368 let mut out = stderr.lock();
369
370 let _ = writeln!(out, "\n {title}\n");
371 for (i, (label, value)) in items.iter().enumerate() {
372 let marker = if i == default_index { ">" } else { " " };
373 let _ = writeln!(out, " {marker} [{num}] {label} {value}", num = i + 1);
374 }
375 let _ = writeln!(out);
376 let _ = write!(out, "Select [1-{}]: ", items.len());
377 let _ = out.flush();
378
379 let mut input = String::new();
380 match std::io::stdin().read_line(&mut input) {
381 Ok(_) => {
382 let input = input.trim();
383 if input.is_empty() {
384 return Some(items[default_index].1.clone());
385 }
386 if let Ok(n) = input.parse::<usize>() {
387 let idx = n.wrapping_sub(1);
388 if idx < items.len() {
389 return Some(items[idx].1.clone());
390 }
391 }
392 None
393 }
394 Err(_) => None,
395 }
396}
397
398#[cfg(test)]
403mod tests {
404 use super::*;
405
406 #[test]
407 fn test_visible_len_plain_text() {
408 assert_eq!(visible_len("hello"), 5);
409 assert_eq!(visible_len(""), 0);
410 assert_eq!(visible_len("abc def"), 7);
411 }
412
413 #[test]
414 fn test_visible_len_with_ansi() {
415 assert_eq!(visible_len("\x1b[1mhello\x1b[0m"), 5);
416 assert_eq!(
417 visible_len("\x1b[1;7m > foo \x1b[0m \x1b[2mbar\x1b[0m"),
418 12
419 );
420 assert_eq!(visible_len("\x1b[32m\x1b[0m"), 0);
421 }
422
423 #[test]
424 fn test_truncate_no_truncation_needed() {
425 let text = "short";
426 assert_eq!(truncate(text, 80), "short");
427 }
428
429 #[test]
430 fn test_truncate_plain_text() {
431 let text = "hello world this is a long string";
432 let result = truncate(text, 10);
433 assert!(visible_len(&result) <= 10);
435 assert!(result.ends_with("\x1b[0m"));
436 }
437
438 #[test]
439 fn test_truncate_with_ansi() {
440 let text = "\x1b[1mhello world long text\x1b[0m";
441 let result = truncate(text, 10);
442 assert!(visible_len(&result) <= 10);
443 assert!(result.ends_with("\x1b[0m"));
444 }
445
446 #[test]
447 fn test_truncate_width_one() {
448 let result = truncate("hello", 1);
449 assert!(result.ends_with("\x1b[0m"));
451 }
452
453 #[test]
454 fn test_arrow_select_empty_items() {
455 assert_eq!(arrow_select(&[], "title", 0), None);
456 }
457
458 #[cfg(unix)]
459 #[test]
460 fn test_key_enum_equality() {
461 assert_eq!(Key::Up, Key::Up);
462 assert_eq!(Key::Number(3), Key::Number(3));
463 assert_ne!(Key::Up, Key::Down);
464 }
465
466 #[test]
467 fn test_fallback_default_index_clamped() {
468 let items = [
469 ("a".to_string(), "val_a".to_string()),
470 ("b".to_string(), "val_b".to_string()),
471 ];
472 let clamped = 10usize.min(items.len() - 1);
473 assert_eq!(clamped, 1);
474 }
475}