use super::theme; use crate::ui::{ component::{Child, Component, ComponentExt, Event, EventCtx}, display, geometry::{Insets, Rect}, }; pub struct Frame { area: Rect, title: U, content: Child, } impl Frame where T: Component, U: AsRef<[u8]>, { pub fn new(area: Rect, title: U, content: impl FnOnce(Rect) -> T) -> Self { let (title_area, content_area) = Self::areas(area); Self { area: title_area, title, content: content(content_area).into_child(), } } fn areas(area: Rect) -> (Rect, Rect) { // Same as PageLayout::BUTTON_SPACE. const TITLE_SPACE: i32 = 6; let (title_area, content_area) = area.split_top(theme::FONT_BOLD.text_height()); let title_area = title_area.inset(Insets::left(theme::CONTENT_BORDER)); let content_area = content_area.inset(Insets::top(TITLE_SPACE)); (title_area, content_area) } } impl Component for Frame where T: Component, U: AsRef<[u8]>, { type Msg = T::Msg; fn event(&mut self, ctx: &mut EventCtx, event: Event) -> Option { self.content.event(ctx, event) } fn paint(&mut self) { display::text( self.area.bottom_left(), self.title.as_ref(), theme::FONT_BOLD, theme::GREY_LIGHT, theme::BG, ); self.content.paint(); } fn bounds(&self, sink: &mut dyn FnMut(Rect)) { sink(self.area); self.content.bounds(sink); } } #[cfg(feature = "ui_debug")] impl crate::trace::Trace for Frame where T: crate::trace::Trace, U: crate::trace::Trace + AsRef<[u8]>, { fn trace(&self, t: &mut dyn crate::trace::Tracer) { t.open("Frame"); t.field("title", &self.title); t.field("content", &self.content); t.close(); } }