use crate::agent::ui::theme::ThemeKey;
use crate::agent::ui::theme::current_theme;
use crate::tui::Component;
use crate::tui::components::r#box::TuiBox;
use crate::tui::components::markdown::{DefaultTextStyle, Markdown};
const OSC133_ZONE_START: &str = "\x1b]133;A\x07";
const OSC133_ZONE_END: &str = "\x1b]133;B\x07";
const OSC133_ZONE_FINAL: &str = "\x1b]133;C\x07";
pub struct UserMessageComponent {
box_component: TuiBox,
cached_lines: Option<Vec<String>>,
cached_width: usize,
}
impl UserMessageComponent {
pub fn new(text: impl Into<String>) -> Self {
let text = text.into();
let theme = current_theme();
let bg_ansi = theme.bg_ansi_key(ThemeKey::UserMessageBg).to_string();
drop(theme);
let mut msg_box = TuiBox::new(1, 1, Some(crate::tui::Style::new().bg(bg_ansi)));
let md_theme = crate::agent::ui::theme::get_markdown_theme();
let default_style = DefaultTextStyle {
color: Some(std::sync::Arc::new(|s: &str| -> String {
let t = current_theme();
t.fg_key(ThemeKey::UserMessageText, s)
})),
bold: false,
italic: false,
strikethrough: false,
underline: false,
};
let md = Markdown::new(text.clone(), 0, 0, md_theme, Some(default_style));
msg_box.add_child(std::boxed::Box::new(md));
Self {
box_component: msg_box,
cached_lines: None,
cached_width: 0,
}
}
}
impl Component for UserMessageComponent {
fn set_expanded(&mut self, _expanded: bool) {
}
fn render(&mut self, width: usize) -> Vec<String> {
if self.cached_width == width
&& let Some(ref lines) = self.cached_lines
{
return lines.clone();
}
let mut lines = self.box_component.render(width);
if !lines.is_empty() {
lines[0] = format!("{}{}", OSC133_ZONE_START, &lines[0]);
if let Some(last) = lines.last_mut() {
last.push_str(OSC133_ZONE_END);
last.push_str(OSC133_ZONE_FINAL);
}
}
let result = lines.clone();
self.cached_lines = Some(lines);
self.cached_width = width;
result
}
fn invalidate(&mut self) {
self.cached_lines = None;
self.box_component.invalidate();
}
}