use ratatui::{
Frame,
layout::Rect,
style::{Color, Modifier, Style},
widgets::Paragraph,
};
use a2ui_base::model::component_context::ComponentContext;
use a2ui_base::protocol::common_types::{DynamicBoolean, DynamicString};
use crate::component_impl::TuiComponent;
pub struct CheckBoxComponent;
impl TuiComponent for CheckBoxComponent {
fn name(&self) -> &'static str {
"CheckBox"
}
fn render(
&self,
ctx: &ComponentContext,
area: Rect,
frame: &mut Frame,
_render_child: &mut dyn FnMut(&str, Rect, &mut Frame, &str),
_measure_child: &mut dyn FnMut(&str, &str, u16) -> Option<u16>,
) {
let comp_model = match ctx.components.get(&ctx.component_id) {
Some(m) => m,
None => return,
};
let inner = crate::layout_engine::padded_content(area);
if inner.width == 0 || inner.height == 0 {
return;
}
let label = match comp_model.get_property::<DynamicString>("label") {
Some(ds) => ctx.data_context.resolve_dynamic_string(&ds),
None => String::new(),
};
let checked = match comp_model.get_property::<DynamicBoolean>("value") {
Some(db) => ctx.data_context.resolve_dynamic_boolean(&db),
None => false,
};
let indicator = if checked { "☑" } else { "☐" };
let display_text = format!("[{}] {}", indicator, label);
let is_focused = ctx.focused_id.as_deref() == Some(ctx.component_id.as_str());
let style = if is_focused {
Style::default().fg(Color::Yellow).add_modifier(Modifier::BOLD)
} else {
Style::default()
};
let paragraph = Paragraph::new(display_text).style(style);
frame.render_widget(paragraph, inner);
}
fn natural_height(
&self,
_ctx: &ComponentContext,
_available_width: u16,
_measure_child: &mut dyn FnMut(&str, &str, u16) -> Option<u16>,
) -> Option<u16> {
Some(3)
}
fn handle_event(
&self,
ctx: &ComponentContext,
event: &a2ui_base::event::InputEvent,
) -> Option<a2ui_base::event::EventResult> {
a2ui_base::components::checkbox::handle_event(ctx, event)
}
}