use crate::ecs::{Entity, World};
pub use crate::components::text_input::{CursorBlinkPhase, cursor_blink_system};
fn attach_handlers_for(world: &mut World, entity: Entity) {
let mut pending: alloc::vec::Vec<crate::widget::view::ViewAttach> = alloc::vec::Vec::new();
if let Some(reg) = world.resource::<crate::widget::view::ViewRegistry>() {
for v in reg.iter() {
if let Some(f) = v.auto_attach() {
pending.push(f);
}
}
}
for f in pending {
f(world, entity);
}
}
pub fn attach_widget_input_handlers(world: &mut World, root: Entity) {
let mut stack = alloc::vec::Vec::with_capacity(16);
stack.push(root);
while let Some(entity) = stack.pop() {
if let Some(children) = world.get::<crate::widget::Children>(entity) {
for &child in &children.0 {
stack.push(child);
}
}
attach_handlers_for(world, entity);
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::components::button::Button;
use crate::components::checkbox::{Checkbox, checkbox_handler};
use crate::components::progress_bar::ProgressBar;
use crate::components::tabbar::{TabBar, tabbar_handler};
use crate::components::text_input::TextInput;
use crate::event::GestureHandler;
use crate::event::gesture::GestureEvent;
use crate::types::{Color, Fixed, Rect};
use crate::widget::ComputedRect;
use crate::widget::view::ViewRegistry;
#[test]
fn tabbar_tap_picks_correct_tab() {
let mut world = World::default();
let e = world.spawn();
world.insert(e, ComputedRect(Rect::new(0, 0, 300, 40)));
world.insert(e, TabBar::new(3));
for (x, expected) in [(50, 0u8), (150, 1), (270, 2), (0, 0), (299, 2)] {
tabbar_handler(
&mut world,
e,
&GestureEvent::Tap {
x: Fixed::from_int(x),
y: Fixed::from_int(20),
target: e,
},
);
let tb = world.get::<TabBar>(e).unwrap();
assert_eq!(tb.selected, expected, "x={x} → expected {expected}");
assert_eq!(tb.indicator_offset, Fixed::from_int(expected as i32));
}
}
#[test]
fn registry_attach_installs_button_gesture_handler() {
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::button::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(e, Button::new(Color::rgb(0, 0, 0), Color::rgb(0, 0, 0)));
attach_handlers_for(&mut world, e);
assert!(
world.get::<GestureHandler>(e).is_some(),
"registry-driven attach must install a GestureHandler on Button entities"
);
}
#[test]
fn registry_attach_skips_when_handler_already_present() {
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::button::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(e, Button::new(Color::rgb(0, 0, 0), Color::rgb(0, 0, 0)));
fn user_handler(_: &mut World, _: Entity, _: &GestureEvent) -> bool {
false
}
world.insert(
e,
GestureHandler {
on_gesture: user_handler,
},
);
attach_handlers_for(&mut world, e);
let h = world.get::<GestureHandler>(e).expect("handler present");
let installed: *const () = h.on_gesture as *const ();
let expected: *const () = user_handler as *const ();
assert!(
core::ptr::eq(installed, expected),
"user-supplied handler must not be overwritten"
);
}
#[test]
fn registry_attach_installs_text_input_handlers_and_focusable() {
use crate::event::focus::{Focusable, KeyHandler};
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::text_input::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(e, TextInput::new());
attach_handlers_for(&mut world, e);
assert!(world.get::<GestureHandler>(e).is_some());
assert!(world.get::<Focusable>(e).is_some());
assert!(world.get::<KeyHandler>(e).is_some());
}
#[test]
fn registry_attach_installs_progress_bar_gesture_handler() {
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::progress_bar::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(
e,
ProgressBar::new(Color::rgb(0, 0, 0), Color::rgb(0, 0, 0)),
);
attach_handlers_for(&mut world, e);
assert!(
world.get::<GestureHandler>(e).is_some(),
"registry-driven attach must install a GestureHandler on ProgressBar entities"
);
}
#[test]
fn registry_attach_installs_checkbox_gesture_handler() {
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::checkbox::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(e, Checkbox::new(Color::rgb(0, 0, 0), Color::rgb(0, 0, 0)));
attach_handlers_for(&mut world, e);
assert!(
world.get::<GestureHandler>(e).is_some(),
"registry-driven attach must install a GestureHandler on Checkbox entities"
);
}
#[test]
fn checkbox_tap_toggles_checked() {
let mut world = World::default();
let mut reg = ViewRegistry::default();
reg.insert(crate::components::checkbox::view());
world.insert_resource(reg);
let e = world.spawn();
world.insert(e, Checkbox::new(Color::rgb(0, 0, 0), Color::rgb(0, 0, 0)));
attach_handlers_for(&mut world, e);
let consumed = checkbox_handler(
&mut world,
e,
&GestureEvent::Tap {
x: Fixed::ZERO,
y: Fixed::ZERO,
target: e,
},
);
assert!(consumed);
assert!(world.get::<Checkbox>(e).unwrap().checked);
}
#[test]
fn tabbar_ignores_non_tap() {
let mut world = World::default();
let e = world.spawn();
world.insert(e, ComputedRect(Rect::new(0, 0, 300, 40)));
world.insert(e, TabBar::new(3));
let consumed = tabbar_handler(
&mut world,
e,
&GestureEvent::DragMove {
x: Fixed::from_int(50),
y: Fixed::from_int(20),
dx: Fixed::ZERO,
dy: Fixed::ZERO,
target: e,
},
);
assert!(!consumed);
assert_eq!(world.get::<TabBar>(e).unwrap().selected, 0);
}
}