use ratatui::{
style::Style,
text::{Line, Span},
};
use unicode_width::UnicodeWidthStr;
pub(crate) fn wrap_line_with_padding<'a>(
line: Line<'a>,
max_width: usize,
padding: &'a str,
) -> Vec<Line<'a>> {
if max_width == 0 {
return vec![line];
}
let total_width: usize = line.spans.iter().map(|s| s.content.width()).sum();
if total_width <= max_width {
return vec![line];
}
let padding_width = padding.width();
let mut segments: Vec<(String, Style)> = Vec::new();
for span in &line.spans {
segments.push((span.content.to_string(), span.style));
}
let mut result: Vec<Line<'a>> = Vec::new();
let mut current_spans: Vec<Span<'a>> = Vec::new();
let mut current_width: usize = 0;
for (text, style) in segments {
let mut remaining = text.as_str();
while !remaining.is_empty() {
let available = max_width.saturating_sub(current_width);
if available == 0 {
result.push(Line::from(current_spans));
current_spans = vec![Span::styled(padding.to_string(), Style::default())];
current_width = padding_width;
continue;
}
let remaining_width = remaining.width();
if remaining_width <= available {
current_spans.push(Span::styled(remaining.to_string(), style));
current_width += remaining_width;
break;
} else {
let byte_limit = char_boundary_at_width(remaining, available);
let break_at = remaining[..byte_limit]
.rfind(' ')
.map(|p| p + 1)
.unwrap_or(byte_limit);
let break_at = if break_at == 0 {
byte_limit.max(remaining.ceil_char_boundary(1))
} else {
break_at
};
let (chunk, rest) = remaining.split_at(break_at);
current_spans.push(Span::styled(chunk.to_string(), style));
remaining = rest.trim_start();
result.push(Line::from(current_spans));
current_spans = vec![Span::styled(padding.to_string(), Style::default())];
current_width = padding_width;
}
}
}
if !current_spans.is_empty() {
result.push(Line::from(current_spans));
}
if result.is_empty() {
result.push(line);
}
result
}
pub(crate) fn char_boundary_at_width(s: &str, target_width: usize) -> usize {
let mut width = 0;
for (idx, ch) in s.char_indices() {
let ch_width = unicode_width::UnicodeWidthChar::width(ch).unwrap_or(0);
if width + ch_width > target_width {
return idx;
}
width += ch_width;
}
s.len()
}
pub(super) fn format_token_count_with_label(tokens: i32, label: &str) -> String {
let tokens = tokens.max(0) as f64;
if tokens >= 1_000_000.0 {
format!("{:.1}M {}", tokens / 1_000_000.0, label)
} else if tokens >= 1_000.0 {
format!("{:.1}K {}", tokens / 1_000.0, label)
} else if tokens > 0.0 {
format!("{} {}", tokens as i32, label)
} else {
"new".to_string()
}
}
pub(super) fn format_token_count_raw(tokens: i32) -> String {
let tokens = tokens.max(0) as f64;
if tokens >= 1_000_000.0 {
format!("{:.1}M", tokens / 1_000_000.0)
} else if tokens >= 1_000.0 {
format!("{:.0}K", tokens / 1_000.0)
} else if tokens > 0.0 {
format!("{}", tokens as i32)
} else {
"0".to_string()
}
}