use super::*;
#[inline]
pub(crate) fn byte_index_for_char(value: &str, char_index: usize) -> usize {
if char_index == 0 {
return 0;
}
value
.char_indices()
.nth(char_index)
.map_or(value.len(), |(idx, _)| idx)
}
#[inline]
pub(crate) fn grapheme_count(s: &str) -> usize {
s.graphemes(true).count()
}
#[inline]
pub(crate) fn byte_index_for_grapheme(s: &str, cluster_index: usize) -> usize {
if cluster_index == 0 {
return 0;
}
s.grapheme_indices(true)
.nth(cluster_index)
.map_or(s.len(), |(idx, _)| idx)
}
#[inline]
pub(crate) fn cluster_width(cluster: &str) -> u32 {
UnicodeWidthStr::width(cluster) as u32
}
pub(crate) fn format_token_count(count: usize) -> String {
if count >= 1_000_000 {
format!("{:.1}M", count as f64 / 1_000_000.0)
} else if count >= 1_000 {
format!("{:.1}k", count as f64 / 1_000.0)
} else {
count.to_string()
}
}
pub(crate) fn format_table_row(cells: &[String], widths: &[u32], separator: &str) -> String {
let sep_width = UnicodeWidthStr::width(separator);
let total_cells_width: usize = widths.iter().map(|w| *w as usize).sum();
let mut row = String::with_capacity(
total_cells_width + sep_width.saturating_mul(widths.len().saturating_sub(1)),
);
for (i, width) in widths.iter().enumerate() {
if i > 0 {
row.push_str(separator);
}
row.push_str(&clamp_table_cell(
cells.get(i).map(String::as_str).unwrap_or(""),
*width,
));
}
row
}
pub(crate) fn clamp_table_cell(cell: &str, width: u32) -> String {
let width = width as usize;
let cell_width = UnicodeWidthStr::width(cell);
if cell_width <= width {
let mut out = String::with_capacity(width);
out.push_str(cell);
out.extend(std::iter::repeat_n(' ', width - cell_width));
return out;
}
if width == 0 {
return String::new();
}
if width == 1 {
return "\u{2026}".to_string();
}
let target = width - 1;
let mut out = String::with_capacity(width);
let mut acc = 0usize;
for ch in cell.chars() {
let ch_width = unicode_width::UnicodeWidthChar::width(ch).unwrap_or(0);
if acc + ch_width > target {
break;
}
out.push(ch);
acc += ch_width;
}
out.push('\u{2026}');
let out_width = UnicodeWidthStr::width(out.as_str());
out.extend(std::iter::repeat_n(' ', width.saturating_sub(out_width)));
out
}
pub(crate) fn table_visible_len(state: &TableState) -> usize {
let visible = state.visible_indices();
if state.page_size == 0 {
return visible.len();
}
let start = state
.page
.saturating_mul(state.page_size)
.min(visible.len());
let end = (start + state.page_size).min(visible.len());
end.saturating_sub(start)
}
pub(crate) fn handle_vertical_nav(
selected: &mut usize,
max_index: usize,
key_code: KeyCode,
) -> bool {
match key_code {
KeyCode::Up | KeyCode::Char('k') if *selected > 0 => {
*selected -= 1;
true
}
KeyCode::Down | KeyCode::Char('j') if *selected < max_index => {
*selected += 1;
true
}
_ => false,
}
}
pub(crate) fn format_compact_number(value: f64) -> String {
if value.fract().abs() < f64::EPSILON {
return format!("{value:.0}");
}
let mut s = format!("{value:.2}");
while s.contains('.') && s.ends_with('0') {
s.pop();
}
if s.ends_with('.') {
s.pop();
}
s
}
pub(crate) fn center_text(text: &str, width: usize) -> String {
let text_width = UnicodeWidthStr::width(text);
if text_width >= width {
return text.to_string();
}
let total = width - text_width;
let left = total / 2;
let right = total - left;
let mut centered = String::with_capacity(width);
centered.extend(std::iter::repeat_n(' ', left));
centered.push_str(text);
centered.extend(std::iter::repeat_n(' ', right));
centered
}
pub(crate) struct TextareaVLine {
pub(crate) logical_row: usize,
pub(crate) char_start: usize,
pub(crate) char_count: usize,
}
pub(crate) fn textarea_build_visual_lines(lines: &[String], wrap_width: u32) -> Vec<TextareaVLine> {
let mut out = Vec::new();
for (row, line) in lines.iter().enumerate() {
if line.is_empty() || wrap_width == u32::MAX {
out.push(TextareaVLine {
logical_row: row,
char_start: 0,
char_count: grapheme_count(line),
});
continue;
}
let mut seg_start = 0usize;
let mut seg_chars = 0usize;
let mut seg_width = 0u32;
for (idx, g) in line.graphemes(true).enumerate() {
let cw = cluster_width(g);
if seg_width + cw > wrap_width && seg_chars > 0 {
out.push(TextareaVLine {
logical_row: row,
char_start: seg_start,
char_count: seg_chars,
});
seg_start = idx;
seg_chars = 0;
seg_width = 0;
}
seg_chars += 1;
seg_width += cw;
}
out.push(TextareaVLine {
logical_row: row,
char_start: seg_start,
char_count: seg_chars,
});
}
out
}
pub(crate) fn textarea_logical_to_visual(
vlines: &[TextareaVLine],
logical_row: usize,
logical_col: usize,
) -> (usize, usize) {
for (i, vl) in vlines.iter().enumerate() {
if vl.logical_row != logical_row {
continue;
}
let seg_end = vl.char_start + vl.char_count;
if logical_col >= vl.char_start && logical_col < seg_end {
return (i, logical_col - vl.char_start);
}
if logical_col == seg_end {
let is_last_seg = vlines
.get(i + 1)
.is_none_or(|next| next.logical_row != logical_row);
if is_last_seg {
return (i, logical_col - vl.char_start);
}
}
}
(vlines.len().saturating_sub(1), 0)
}
pub(crate) fn textarea_visual_to_logical(
vlines: &[TextareaVLine],
visual_row: usize,
visual_col: usize,
) -> (usize, usize) {
if let Some(vl) = vlines.get(visual_row) {
let logical_col = vl.char_start + visual_col.min(vl.char_count);
(vl.logical_row, logical_col)
} else {
(0, 0)
}
}
impl Context {
pub fn measure_text(&self, text: &str, max_width: Option<u16>) -> (u16, u16) {
let budget = match max_width {
Some(w) if w > 0 => w as u32,
_ => u32::MAX,
};
let lines = crate::layout::wrap_lines(text, budget);
let height = lines.len().max(1);
let width = lines
.iter()
.map(|line| UnicodeWidthStr::width(line.as_str()))
.max()
.unwrap_or(0);
(clamp_u16(width), clamp_u16(height))
}
pub fn measured_rect(&self, name: &str) -> Option<Rect> {
self.prev_group_rects
.iter()
.find(|(group_name, _)| group_name.as_ref() == name)
.map(|(_, rect)| *rect)
}
}
#[inline]
fn clamp_u16(value: usize) -> u16 {
value.min(u16::MAX as usize) as u16
}
#[allow(unused_variables)]
pub(crate) fn open_url(url: &str) -> std::io::Result<()> {
#[cfg(target_os = "macos")]
{
std::process::Command::new("open").arg(url).spawn()?;
}
#[cfg(target_os = "linux")]
{
std::process::Command::new("xdg-open").arg(url).spawn()?;
}
#[cfg(target_os = "windows")]
{
std::process::Command::new("cmd")
.args(["/c", "start", "", url])
.spawn()?;
}
Ok(())
}
#[cfg(test)]
mod measure_tests {
use crate::test_utils::TestBackend;
use crate::{Border, Context, FrameState, Theme};
#[test]
fn measure_text_unwrapped_reports_widest_line_and_line_count() {
let mut state = FrameState::default();
let ui = Context::new(Vec::new(), 40, 10, &mut state, Theme::dark());
let (w, h) = ui.measure_text("hello\nworld!", None);
assert_eq!((w, h), (6, 2));
assert_eq!(ui.measure_text("abc", None), (3, 1));
assert_eq!(ui.measure_text("", None), (0, 1));
}
#[test]
fn measure_text_wraps_to_budget_and_never_exceeds_it() {
let mut state = FrameState::default();
let ui = Context::new(Vec::new(), 40, 10, &mut state, Theme::dark());
let (w, h) = ui.measure_text("alpha beta gamma", Some(5));
assert!(w <= 5, "wrapped width {w} must not exceed the budget");
assert_eq!(h, 3, "three 5-wide words wrap onto three rows");
assert_eq!(w, 5);
let (w, h) = ui.measure_text("abcdefghij", Some(4));
assert!(w <= 4);
assert!(h >= 3, "10 chars at width 4 need at least 3 rows, got {h}");
}
#[test]
fn measure_text_some_zero_is_treated_as_unbounded() {
let mut state = FrameState::default();
let ui = Context::new(Vec::new(), 40, 10, &mut state, Theme::dark());
assert_eq!(
ui.measure_text("a b c\nlonger line", Some(0)),
ui.measure_text("a b c\nlonger line", None),
);
}
#[test]
fn measure_text_counts_wide_cjk_glyphs_as_two_cells() {
let mut state = FrameState::default();
let ui = Context::new(Vec::new(), 40, 10, &mut state, Theme::dark());
assert_eq!(ui.measure_text("νκΈ", None), (4, 1));
}
#[test]
fn measured_rect_is_none_on_first_frame() {
let mut state = FrameState::default();
let ui = Context::new(Vec::new(), 40, 10, &mut state, Theme::dark());
assert!(ui.measured_rect("panel").is_none());
}
#[test]
fn measured_rect_returns_group_geometry_after_a_render() {
let mut backend = TestBackend::new(40, 10);
backend.render(|ui| {
let _ = ui.group("panel").border(Border::Rounded).col(|ui| {
ui.text("hi");
});
});
let mut seen: Option<crate::Rect> = None;
backend.render(|ui| {
seen = ui.measured_rect("panel");
assert!(ui.measured_rect("does-not-exist").is_none());
});
let rect = seen.expect("named group must have a measured rect after render");
assert!(
rect.width > 0 && rect.height > 0,
"measured rect must be non-empty, got {rect:?}"
);
assert!(rect.x + rect.width <= 40);
assert!(rect.y + rect.height <= 10);
}
}