use crate::core::{Color, Font, Point, Rect};
use crate::event::{Event, EventHandler};
use crate::render::RenderContext;
use crate::signal::Signal1;
use crate::widget::{BaseWidget, Draw, Widget, WidgetKind};
pub struct EditableComboBox {
base: BaseWidget,
text: String,
items: Vec<String>,
expanded: bool,
selected_index: Option<usize>,
pub text_changed: Signal1<String>,
pub item_selected: Signal1<usize>,
}
impl EditableComboBox {
pub fn new(geometry: Rect) -> Self {
Self {
base: BaseWidget::new(WidgetKind::EditableComboBox, geometry, "EditableComboBox"),
text: String::new(),
items: Vec::new(),
expanded: false,
selected_index: None,
text_changed: Signal1::new(),
item_selected: Signal1::new(),
}
}
pub fn text(&self) -> &str {
&self.text
}
pub fn set_text(&mut self, text: impl Into<String>) {
let t = text.into();
if self.text != t {
self.text = t.clone();
self.text_changed.emit(self.text.clone());
self.base.request_redraw();
}
}
pub fn add_item(&mut self, item: impl Into<String>) {
self.items.push(item.into());
self.base.request_redraw();
}
pub fn remove_item(&mut self, index: usize) -> bool {
if index < self.items.len() {
self.items.remove(index);
if self.selected_index == Some(index) {
self.selected_index = None;
} else if let Some(ref mut sel) = self.selected_index {
if index < *sel {
*sel -= 1;
}
}
self.base.request_redraw();
true
} else {
false
}
}
pub fn clear_items(&mut self) {
self.items.clear();
self.selected_index = None;
self.expanded = false;
self.base.request_redraw();
}
pub fn items(&self) -> &[String] {
&self.items
}
pub fn item_count(&self) -> usize {
self.items.len()
}
pub fn is_expanded(&self) -> bool {
self.expanded
}
pub fn expand(&mut self) {
if !self.expanded {
self.expanded = true;
self.base.request_redraw();
}
}
pub fn collapse(&mut self) {
if self.expanded {
self.expanded = false;
self.base.request_redraw();
}
}
pub fn toggle(&mut self) {
self.expanded = !self.expanded;
self.base.request_redraw();
}
pub fn selected_index(&self) -> Option<usize> {
self.selected_index
}
pub fn select_index(&mut self, index: usize) -> bool {
if index < self.items.len() {
if self.selected_index != Some(index) {
self.selected_index = Some(index);
self.text = self.items[index].clone();
self.text_changed.emit(self.text.clone());
self.item_selected.emit(index);
self.expanded = false;
self.base.request_redraw();
return true;
}
}
false
}
}
impl Widget for EditableComboBox {
fn base(&self) -> &BaseWidget {
&self.base
}
fn base_mut(&mut self) -> &mut BaseWidget {
&mut self.base
}
}
impl Draw for EditableComboBox {
fn draw(&mut self, context: &mut RenderContext) {
let rect = self.geometry();
let is_enabled = self.base.is_enabled();
let bg_color = if is_enabled {
Color::rgba(255, 255, 255, 255)
} else {
Color::rgba(240, 240, 240, 255)
};
context.fill_rounded_rect(rect, 4, bg_color);
let border_color = if is_enabled {
Color::rgba(180, 180, 180, 255)
} else {
Color::rgba(210, 210, 210, 255)
};
context.draw_rounded_rect_stroke(rect, 4, border_color, 1);
let font = Font::simple("sans-serif", 13.0);
let padding = 6i32;
let text_x = rect.x + padding;
let text_y = rect.y + padding + 13;
let display_text = if self.text.is_empty() && !is_enabled { "" } else { &self.text };
let text_color =
if is_enabled { Color::rgba(0, 0, 0, 255) } else { Color::rgba(180, 180, 180, 255) };
context.draw_text(Point::new(text_x, text_y), display_text, &font, text_color);
let arrow_x = rect.x + rect.width as i32 - 20;
let arrow_y = rect.y + rect.height as i32 / 2 - 2;
let arrow_color = if is_enabled {
Color::rgba(100, 100, 100, 255)
} else {
Color::rgba(180, 180, 180, 255)
};
context.draw_text(
Point::new(arrow_x, arrow_y),
if self.expanded { "▲" } else { "▼" },
&font,
arrow_color,
);
if !self.expanded || self.items.is_empty() {
return;
}
let drop_down_y = rect.y + rect.height as i32;
let item_height = 28u32;
let drop_down_height = item_height * self.items.len() as u32;
let drop_rect = Rect::new(rect.x, drop_down_y, rect.width, drop_down_height);
context.fill_rounded_rect(drop_rect, 2, Color::rgba(255, 255, 255, 255));
context.draw_rounded_rect_stroke(drop_rect, 2, Color::rgba(200, 200, 200, 255), 1);
for (i, item) in self.items.iter().enumerate() {
let item_rect = Rect::new(
rect.x + 1,
drop_down_y + (i as i32) * (item_height as i32),
rect.width - 2,
item_height,
);
if self.selected_index == Some(i) {
context.fill_rounded_rect(item_rect, 2, Color::rgba(52, 120, 246, 40));
}
let item_text_x = item_rect.x + 8;
let item_text_y = item_rect.y + 18;
let item_color = if is_enabled {
Color::rgba(0, 0, 0, 255)
} else {
Color::rgba(180, 180, 180, 255)
};
context.draw_text(Point::new(item_text_x, item_text_y), item, &font, item_color);
}
}
}
impl EventHandler for EditableComboBox {
fn handle_event(&mut self, event: &Event) {
if !self.base.is_enabled() {
return;
}
match event {
Event::KeyPress { key, .. } => {
match *key {
8 => {
let mut new_text = self.text.clone();
new_text.pop();
self.set_text(new_text);
return;
}
13 => {
if self.expanded && !self.items.is_empty() {
if let Some(idx) = self.selected_index {
self.select_index(idx);
} else if let Some(idx) =
self.items.iter().position(|item| item == &self.text)
{
self.select_index(idx);
}
}
self.expanded = false;
self.base.request_redraw();
return;
}
27 => {
self.expanded = false;
self.base.request_redraw();
return;
}
38 => {
if self.expanded && !self.items.is_empty() {
let prev = match self.selected_index {
Some(0) | None => 0,
Some(idx) => idx - 1,
};
self.selected_index = Some(prev);
self.base.request_redraw();
}
return;
}
40 => {
if !self.expanded {
self.expanded = true;
}
if !self.items.is_empty() {
let next = match self.selected_index {
Some(idx) => (idx + 1).min(self.items.len() - 1),
None => 0,
};
self.selected_index = Some(next);
self.base.request_redraw();
}
return;
}
_ => {
if let Some(ch) = char::from_u32(*key) {
if ch.is_ascii_graphic() || ch == ' ' {
let mut new_text = self.text.clone();
new_text.push(ch);
self.set_text(new_text);
return;
}
}
}
}
self.base.handle_event(event);
}
Event::MousePress { pos, button } if *button == 1 => {
let rect = self.geometry();
if rect.contains_point(*pos) {
let arrow_zone_x = rect.x + rect.width as i32 - 24;
if pos.x >= arrow_zone_x {
self.toggle();
} else {
if !self.expanded && !self.items.is_empty() {
self.expand();
} else {
self.toggle();
}
}
return;
}
if self.expanded {
let item_height = 28i32;
let drop_down_y = rect.y + rect.height as i32;
let drop_down_height = item_height * self.items.len() as i32;
let drop_rect =
Rect::new(rect.x, drop_down_y, rect.width, drop_down_height as u32);
if drop_rect.contains_point(*pos) {
let rel_y = pos.y - drop_down_y;
let idx = (rel_y / item_height) as usize;
if idx < self.items.len() {
self.select_index(idx);
}
return;
}
if !rect.contains_point(*pos) {
self.collapse();
return;
}
}
}
_ => {
self.base.handle_event(event);
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::widget::svg::render_to_svg;
#[test]
fn editable_combo_box_default_creation() {
let cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
assert_eq!(cb.kind(), WidgetKind::EditableComboBox);
assert_eq!(cb.text(), "");
assert_eq!(cb.item_count(), 0);
assert!(!cb.is_expanded());
assert!(cb.selected_index().is_none());
}
#[test]
fn editable_combo_box_add_and_remove_items() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
cb.add_item("Option A");
cb.add_item("Option B");
cb.add_item("Option C");
assert_eq!(cb.item_count(), 3);
assert!(cb.remove_item(1));
assert_eq!(cb.item_count(), 2);
assert_eq!(cb.items()[0], "Option A");
assert_eq!(cb.items()[1], "Option C");
assert!(!cb.remove_item(5));
assert_eq!(cb.item_count(), 2);
}
#[test]
fn editable_combo_box_text_operations() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
assert_eq!(cb.text(), "");
cb.set_text("Hello");
assert_eq!(cb.text(), "Hello");
cb.set_text("World");
assert_eq!(cb.text(), "World");
cb.clear_items();
assert_eq!(cb.text(), "World");
}
#[test]
fn editable_combo_box_expand_collapse() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
assert!(!cb.is_expanded());
cb.expand();
assert!(cb.is_expanded());
cb.collapse();
assert!(!cb.is_expanded());
cb.toggle();
assert!(cb.is_expanded());
cb.toggle();
assert!(!cb.is_expanded());
}
#[test]
fn editable_combo_box_select_index() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
cb.add_item("Alpha");
cb.add_item("Beta");
cb.add_item("Gamma");
assert!(cb.select_index(1));
assert_eq!(cb.selected_index(), Some(1));
assert_eq!(cb.text(), "Beta");
assert!(!cb.is_expanded());
assert!(!cb.select_index(1));
assert!(!cb.select_index(10));
}
#[test]
fn editable_combo_box_svg_output() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
cb.add_item("Item 1");
cb.add_item("Item 2");
cb.set_text("Hello");
let svg = render_to_svg(&mut cb);
assert!(svg.starts_with("<svg"));
assert!(svg.ends_with("</svg>"));
}
#[test]
fn editable_combo_box_text_changed_signal() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
let captured = std::sync::Arc::new(std::sync::Mutex::new(None::<String>));
let c = captured.clone();
cb.text_changed.connect(move |val: std::sync::Arc<String>| {
*c.lock().unwrap() = Some(val.to_string());
});
cb.set_text("CustomValue");
assert_eq!(captured.lock().unwrap().as_deref(), Some("CustomValue"));
}
#[test]
fn editable_combo_box_item_selected_signal() {
let mut cb = EditableComboBox::new(Rect::new(0, 0, 200, 30));
cb.add_item("A");
cb.add_item("B");
let captured = std::sync::Arc::new(std::sync::Mutex::new(None::<usize>));
let c = captured.clone();
cb.item_selected.connect(move |val: std::sync::Arc<usize>| {
*c.lock().unwrap() = Some(*val);
});
cb.select_index(1);
assert_eq!(*captured.lock().unwrap(), Some(1));
}
}