use crate::core::{ObjectId, Rect};
use crate::render::RenderContext;
use crate::signal::GenericSignal;
use crate::widget::{BaseWidget, Draw, Widget, WidgetKind};
pub struct PopupWindow {
base: BaseWidget,
content_widget: Option<ObjectId>,
pub opened: GenericSignal,
pub closed: GenericSignal,
}
impl PopupWindow {
pub fn new(geometry: Rect) -> Self {
Self {
base: BaseWidget::new(WidgetKind::PopupWindow, geometry, "PopupWindow"),
content_widget: None,
opened: GenericSignal::new(),
closed: GenericSignal::new(),
}
}
pub fn content_widget(&self) -> Option<ObjectId> {
self.content_widget
}
pub fn set_content_widget(&mut self, widget: Option<ObjectId>) {
if let Some(old) = self.content_widget {
self.base.remove_child(old);
}
self.content_widget = widget;
if let Some(id) = widget {
self.base.add_child(id);
}
self.base.request_redraw();
}
pub fn open(&mut self) {
self.show();
self.opened.emit();
}
pub fn close(&mut self) {
self.hide();
self.closed.emit();
}
}
impl Widget for PopupWindow {
fn base(&self) -> &BaseWidget {
&self.base
}
fn base_mut(&mut self) -> &mut BaseWidget {
&mut self.base
}
}
impl Draw for PopupWindow {
fn draw(&mut self, context: &mut RenderContext) {
let rect = self.base.geometry();
use crate::core::Color;
context.fill_rect(rect, Color::from_rgb(255, 255, 255));
context.draw_rect(rect, Color::from_rgb(120, 120, 120));
}
}
impl crate::event::EventHandler for PopupWindow {
fn handle_event(&mut self, event: &crate::event::Event) {
if !self.base.is_enabled() {
return;
}
match event {
crate::event::Event::MousePress { pos: _, button } if *button == 1 => {
self.base.set_mouse_pressed(true);
}
crate::event::Event::MouseRelease { pos: _, button } if *button == 1 => {
self.base.set_mouse_pressed(false);
}
_ => { }
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::object::Object;
use std::sync::{Arc, Mutex};
#[test]
fn popup_open_close_emits_lifecycle_signals() {
let mut popup = PopupWindow::new(Rect::new(0, 0, 120, 80));
let opened = Arc::new(Mutex::new(0usize));
let closed = Arc::new(Mutex::new(0usize));
let opened_sink = opened.clone();
popup.opened.connect(move || {
if let Ok(mut count) = opened_sink.lock() {
*count += 1;
}
});
let closed_sink = closed.clone();
popup.closed.connect(move || {
if let Ok(mut count) = closed_sink.lock() {
*count += 1;
}
});
popup.open();
popup.close();
assert!(!popup.is_visible());
assert_eq!(*opened.lock().expect("opened lock poisoned"), 1);
assert_eq!(*closed.lock().expect("closed lock poisoned"), 1);
}
#[test]
fn popup_replaces_content_widget_child_binding() {
let mut popup = PopupWindow::new(Rect::new(0, 0, 120, 80));
let old_id = Object::new("OldContent").id();
let new_id = Object::new("NewContent").id();
popup.set_content_widget(Some(old_id));
assert_eq!(popup.content_widget(), Some(old_id));
assert_eq!(popup.children(), &[old_id]);
popup.set_content_widget(Some(new_id));
assert_eq!(popup.content_widget(), Some(new_id));
assert_eq!(popup.children(), &[new_id]);
}
}