mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-24 22:31:35 +00:00
refactor(core/rust/ui): simplify button height computation
[no changelog]
This commit is contained in:
parent
885ae2a943
commit
5052594789
@ -23,7 +23,7 @@ pub use maybe::Maybe;
|
|||||||
pub use pad::Pad;
|
pub use pad::Pad;
|
||||||
pub use paginated::{PageMsg, Paginate};
|
pub use paginated::{PageMsg, Paginate};
|
||||||
pub use painter::{qrcode_painter, Painter};
|
pub use painter::{qrcode_painter, Painter};
|
||||||
pub use placed::GridPlaced;
|
pub use placed::{FixedHeightBar, GridPlaced};
|
||||||
pub use text::{
|
pub use text::{
|
||||||
formatted::FormattedText,
|
formatted::FormattedText,
|
||||||
layout::{LineBreaking, PageBreaking, TextLayout},
|
layout::{LineBreaking, PageBreaking, TextLayout},
|
||||||
|
@ -77,3 +77,47 @@ where
|
|||||||
d.close();
|
d.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct FixedHeightBar<T> {
|
||||||
|
inner: T,
|
||||||
|
height: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> FixedHeightBar<T> {
|
||||||
|
pub const fn bottom(inner: T, height: i32) -> Self {
|
||||||
|
Self { inner, height }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Component for FixedHeightBar<T>
|
||||||
|
where
|
||||||
|
T: Component,
|
||||||
|
{
|
||||||
|
type Msg = T::Msg;
|
||||||
|
|
||||||
|
fn place(&mut self, bounds: Rect) -> Rect {
|
||||||
|
let (_, bar) = bounds.split_bottom(self.height);
|
||||||
|
self.inner.place(bar)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn event(&mut self, ctx: &mut EventCtx, event: Event) -> Option<Self::Msg> {
|
||||||
|
self.inner.event(ctx, event)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn paint(&mut self) {
|
||||||
|
self.inner.paint()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "ui_debug")]
|
||||||
|
impl<T> crate::trace::Trace for FixedHeightBar<T>
|
||||||
|
where
|
||||||
|
T: Component,
|
||||||
|
T: crate::trace::Trace,
|
||||||
|
{
|
||||||
|
fn trace(&self, d: &mut dyn crate::trace::Tracer) {
|
||||||
|
d.open("FixedHeightBar");
|
||||||
|
d.field("inner", &self.inner);
|
||||||
|
d.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
time::Duration,
|
time::Duration,
|
||||||
ui::{
|
ui::{
|
||||||
component::{Component, ComponentExt, Event, EventCtx, GridPlaced, Map, TimerToken},
|
component::{
|
||||||
|
Component, ComponentExt, Event, EventCtx, FixedHeightBar, GridPlaced, Map, TimerToken,
|
||||||
|
},
|
||||||
display::{self, Color, Font},
|
display::{self, Color, Font},
|
||||||
event::TouchEvent,
|
event::TouchEvent,
|
||||||
geometry::{Insets, Offset, Rect},
|
geometry::{Insets, Offset, Rect},
|
||||||
@ -27,9 +29,6 @@ pub struct Button<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<T> Button<T> {
|
impl<T> Button<T> {
|
||||||
/// Standard height in pixels.
|
|
||||||
pub const HEIGHT: i32 = 38;
|
|
||||||
|
|
||||||
/// Offsets the baseline of the button text either up (negative) or down
|
/// Offsets the baseline of the button text either up (negative) or down
|
||||||
/// (positive).
|
/// (positive).
|
||||||
pub const BASELINE_OFFSET: i32 = -3;
|
pub const BASELINE_OFFSET: i32 = -3;
|
||||||
@ -353,7 +352,7 @@ impl<T> Button<T> {
|
|||||||
T: AsRef<str>,
|
T: AsRef<str>,
|
||||||
{
|
{
|
||||||
let columns = 1 + right_size_factor;
|
let columns = 1 + right_size_factor;
|
||||||
(
|
theme::button_bar((
|
||||||
GridPlaced::new(left)
|
GridPlaced::new(left)
|
||||||
.with_grid(1, columns)
|
.with_grid(1, columns)
|
||||||
.with_spacing(theme::BUTTON_SPACING)
|
.with_spacing(theme::BUTTON_SPACING)
|
||||||
@ -368,7 +367,7 @@ impl<T> Button<T> {
|
|||||||
.map(|msg| {
|
.map(|msg| {
|
||||||
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
||||||
}),
|
}),
|
||||||
)
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn cancel_confirm_text(
|
pub fn cancel_confirm_text(
|
||||||
@ -407,6 +406,8 @@ impl<T> Button<T> {
|
|||||||
let right = Button::with_text(confirm).styled(theme::button_confirm());
|
let right = Button::with_text(confirm).styled(theme::button_confirm());
|
||||||
let top = Button::with_text(info);
|
let top = Button::with_text(info);
|
||||||
let left = Button::with_icon(theme::ICON_CANCEL);
|
let left = Button::with_icon(theme::ICON_CANCEL);
|
||||||
|
theme::button_bar_rows(
|
||||||
|
2,
|
||||||
(
|
(
|
||||||
GridPlaced::new(left)
|
GridPlaced::new(left)
|
||||||
.with_grid(2, 3)
|
.with_grid(2, 3)
|
||||||
@ -419,7 +420,9 @@ impl<T> Button<T> {
|
|||||||
.with_grid(2, 3)
|
.with_grid(2, 3)
|
||||||
.with_spacing(theme::BUTTON_SPACING)
|
.with_spacing(theme::BUTTON_SPACING)
|
||||||
.with_from_to((0, 0), (0, 2))
|
.with_from_to((0, 0), (0, 2))
|
||||||
.map(|msg| (matches!(msg, ButtonMsg::Clicked)).then(|| CancelInfoConfirmMsg::Info)),
|
.map(|msg| {
|
||||||
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelInfoConfirmMsg::Info)
|
||||||
|
}),
|
||||||
GridPlaced::new(right)
|
GridPlaced::new(right)
|
||||||
.with_grid(2, 3)
|
.with_grid(2, 3)
|
||||||
.with_spacing(theme::BUTTON_SPACING)
|
.with_spacing(theme::BUTTON_SPACING)
|
||||||
@ -427,6 +430,7 @@ impl<T> Button<T> {
|
|||||||
.map(|msg| {
|
.map(|msg| {
|
||||||
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelInfoConfirmMsg::Confirmed)
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelInfoConfirmMsg::Confirmed)
|
||||||
}),
|
}),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -452,25 +456,25 @@ impl<T> Button<T> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let [top, middle, bottom] = words;
|
let [top, middle, bottom] = words;
|
||||||
(btn(0, top), btn(1, middle), btn(2, bottom))
|
theme::button_bar_rows(3, (btn(0, top), btn(1, middle), btn(2, bottom)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type CancelConfirm<T, F0, F1> = (
|
type CancelConfirm<T, F0, F1> = FixedHeightBar<(
|
||||||
Map<GridPlaced<Button<T>>, F0>,
|
Map<GridPlaced<Button<T>>, F0>,
|
||||||
Map<GridPlaced<Button<T>>, F1>,
|
Map<GridPlaced<Button<T>>, F1>,
|
||||||
);
|
)>;
|
||||||
|
|
||||||
pub enum CancelConfirmMsg {
|
pub enum CancelConfirmMsg {
|
||||||
Cancelled,
|
Cancelled,
|
||||||
Confirmed,
|
Confirmed,
|
||||||
}
|
}
|
||||||
|
|
||||||
type CancelInfoConfirm<T, F0, F1, F2> = (
|
type CancelInfoConfirm<T, F0, F1, F2> = FixedHeightBar<(
|
||||||
Map<GridPlaced<Button<T>>, F0>,
|
Map<GridPlaced<Button<T>>, F0>,
|
||||||
Map<GridPlaced<Button<T>>, F1>,
|
Map<GridPlaced<Button<T>>, F1>,
|
||||||
Map<GridPlaced<Button<T>>, F2>,
|
Map<GridPlaced<Button<T>>, F2>,
|
||||||
);
|
)>;
|
||||||
|
|
||||||
pub enum CancelInfoConfirmMsg {
|
pub enum CancelInfoConfirmMsg {
|
||||||
Cancelled,
|
Cancelled,
|
||||||
|
@ -3,7 +3,7 @@ use crate::ui::{
|
|||||||
geometry::{Insets, LinearPlacement, Rect},
|
geometry::{Insets, LinearPlacement, Rect},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{theme, Button};
|
use super::theme;
|
||||||
|
|
||||||
pub enum DialogMsg<T, U> {
|
pub enum DialogMsg<T, U> {
|
||||||
Content(T),
|
Content(T),
|
||||||
@ -40,9 +40,12 @@ where
|
|||||||
type Msg = DialogMsg<T::Msg, U::Msg>;
|
type Msg = DialogMsg<T::Msg, U::Msg>;
|
||||||
|
|
||||||
fn place(&mut self, bounds: Rect) -> Rect {
|
fn place(&mut self, bounds: Rect) -> Rect {
|
||||||
let layout = DialogLayout::middle(bounds);
|
let controls_area = self.controls.place(bounds);
|
||||||
self.content.place(layout.content);
|
let content_area = bounds
|
||||||
self.controls.place(layout.controls);
|
.inset(Insets::bottom(controls_area.height()))
|
||||||
|
.inset(Insets::bottom(theme::BUTTON_SPACING))
|
||||||
|
.inset(Insets::left(theme::CONTENT_BORDER));
|
||||||
|
self.content.place(content_area);
|
||||||
bounds
|
bounds
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -64,21 +67,6 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct DialogLayout {
|
|
||||||
pub content: Rect,
|
|
||||||
pub controls: Rect,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DialogLayout {
|
|
||||||
pub fn middle(area: Rect) -> Self {
|
|
||||||
let (content, controls) = area.split_bottom(Button::<&str>::HEIGHT);
|
|
||||||
let content = content
|
|
||||||
.inset(Insets::bottom(theme::BUTTON_SPACING))
|
|
||||||
.inset(Insets::left(theme::CONTENT_BORDER));
|
|
||||||
Self { content, controls }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "ui_debug")]
|
#[cfg(feature = "ui_debug")]
|
||||||
impl<T, U> crate::trace::Trace for Dialog<T, U>
|
impl<T, U> crate::trace::Trace for Dialog<T, U>
|
||||||
where
|
where
|
||||||
@ -145,12 +133,14 @@ where
|
|||||||
let bounds = bounds
|
let bounds = bounds
|
||||||
.inset(theme::borders())
|
.inset(theme::borders())
|
||||||
.inset(Insets::top(Self::ICON_AREA_PADDING));
|
.inset(Insets::top(Self::ICON_AREA_PADDING));
|
||||||
let (content, buttons) = bounds.split_bottom(Button::<&str>::HEIGHT);
|
|
||||||
let (image, content) = content.split_top(Self::ICON_AREA_HEIGHT);
|
|
||||||
|
|
||||||
self.image.place(image);
|
let controls_area = self.controls.place(bounds);
|
||||||
self.paragraphs.place(content);
|
let content_area = bounds.inset(Insets::bottom(controls_area.height()));
|
||||||
self.controls.place(buttons);
|
|
||||||
|
let (image_area, content_area) = content_area.split_top(Self::ICON_AREA_HEIGHT);
|
||||||
|
|
||||||
|
self.image.place(image_area);
|
||||||
|
self.paragraphs.place(content_area);
|
||||||
bounds
|
bounds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
time::Instant,
|
time::Instant,
|
||||||
ui::{
|
ui::{
|
||||||
component::{Child, Component, ComponentExt, Event, EventCtx, Pad},
|
component::{Child, Component, ComponentExt, Event, EventCtx, FixedHeightBar, Pad},
|
||||||
geometry::{Grid, Rect},
|
geometry::{Grid, Insets, Rect},
|
||||||
model_tt::component::DialogLayout,
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -18,7 +17,7 @@ pub enum HoldToConfirmMsg<T> {
|
|||||||
pub struct HoldToConfirm<T> {
|
pub struct HoldToConfirm<T> {
|
||||||
loader: Loader,
|
loader: Loader,
|
||||||
content: Child<T>,
|
content: Child<T>,
|
||||||
buttons: CancelHold,
|
buttons: FixedHeightBar<CancelHold>,
|
||||||
pad: Pad,
|
pad: Pad,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -47,11 +46,14 @@ where
|
|||||||
type Msg = HoldToConfirmMsg<T::Msg>;
|
type Msg = HoldToConfirmMsg<T::Msg>;
|
||||||
|
|
||||||
fn place(&mut self, bounds: Rect) -> Rect {
|
fn place(&mut self, bounds: Rect) -> Rect {
|
||||||
let layout = DialogLayout::middle(bounds);
|
let controls_area = self.buttons.place(bounds);
|
||||||
self.pad.place(layout.content);
|
let content_area = bounds
|
||||||
self.loader.place(layout.content);
|
.inset(Insets::bottom(controls_area.height()))
|
||||||
self.content.place(layout.content);
|
.inset(Insets::bottom(theme::BUTTON_SPACING))
|
||||||
self.buttons.place(layout.controls);
|
.inset(Insets::left(theme::CONTENT_BORDER));
|
||||||
|
self.pad.place(content_area);
|
||||||
|
self.loader.place(content_area);
|
||||||
|
self.content.place(content_area);
|
||||||
bounds
|
bounds
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -121,18 +123,18 @@ pub enum CancelHoldMsg {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl CancelHold {
|
impl CancelHold {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> FixedHeightBar<Self> {
|
||||||
Self {
|
theme::button_bar(Self {
|
||||||
cancel: Some(Button::with_icon(theme::ICON_CANCEL)),
|
cancel: Some(Button::with_icon(theme::ICON_CANCEL)),
|
||||||
hold: Button::with_text("HOLD TO CONFIRM").styled(theme::button_confirm()),
|
hold: Button::with_text("HOLD TO CONFIRM").styled(theme::button_confirm()),
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn without_cancel() -> Self {
|
pub fn without_cancel() -> FixedHeightBar<Self> {
|
||||||
Self {
|
theme::button_bar(Self {
|
||||||
cancel: None,
|
cancel: None,
|
||||||
hold: Button::with_text("HOLD TO CONFIRM").styled(theme::button_confirm()),
|
hold: Button::with_text("HOLD TO CONFIRM").styled(theme::button_confirm()),
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,7 +71,7 @@ where
|
|||||||
|
|
||||||
fn place(&mut self, bounds: Rect) -> Rect {
|
fn place(&mut self, bounds: Rect) -> Rect {
|
||||||
self.area = bounds;
|
self.area = bounds;
|
||||||
let button_height = Button::<&str>::HEIGHT;
|
let button_height = theme::BUTTON_HEIGHT;
|
||||||
let content_area = self.area.inset(Insets::top(2 * theme::BUTTON_SPACING));
|
let content_area = self.area.inset(Insets::top(2 * theme::BUTTON_SPACING));
|
||||||
let (input_area, content_area) = content_area.split_top(button_height);
|
let (input_area, content_area) = content_area.split_top(button_height);
|
||||||
let (content_area, button_area) = content_area.split_bottom(button_height);
|
let (content_area, button_area) = content_area.split_bottom(button_height);
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
use crate::ui::{
|
use crate::ui::{
|
||||||
component::{
|
component::{
|
||||||
base::ComponentExt, paginated::PageMsg, Component, Event, EventCtx, Label, Pad, Paginate,
|
base::ComponentExt, paginated::PageMsg, Component, Event, EventCtx, FixedHeightBar, Label,
|
||||||
|
Pad, Paginate,
|
||||||
},
|
},
|
||||||
display::{self, Color},
|
display::{self, Color},
|
||||||
geometry::Rect,
|
geometry::{Insets, Rect},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
hold_to_confirm::{handle_hold_event, CancelHold, CancelHoldMsg},
|
hold_to_confirm::{handle_hold_event, CancelHold, CancelHoldMsg},
|
||||||
theme, Button, CancelConfirmMsg, Loader, ScrollBar, Swipe, SwipeDirection,
|
theme, CancelConfirmMsg, Loader, ScrollBar, Swipe, SwipeDirection,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct SwipePage<T, U> {
|
pub struct SwipePage<T, U> {
|
||||||
@ -19,7 +20,6 @@ pub struct SwipePage<T, U> {
|
|||||||
scrollbar: ScrollBar,
|
scrollbar: ScrollBar,
|
||||||
hint: Label<&'static str>,
|
hint: Label<&'static str>,
|
||||||
fade: Option<i32>,
|
fade: Option<i32>,
|
||||||
button_rows: i32,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T, U> SwipePage<T, U>
|
impl<T, U> SwipePage<T, U>
|
||||||
@ -37,15 +37,9 @@ where
|
|||||||
pad: Pad::with_background(background),
|
pad: Pad::with_background(background),
|
||||||
hint: Label::centered("SWIPE TO CONTINUE", theme::label_page_hint()),
|
hint: Label::centered("SWIPE TO CONTINUE", theme::label_page_hint()),
|
||||||
fade: None,
|
fade: None,
|
||||||
button_rows: 1,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_button_rows(mut self, rows: usize) -> Self {
|
|
||||||
self.button_rows = rows as i32;
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
fn setup_swipe(&mut self) {
|
fn setup_swipe(&mut self) {
|
||||||
self.swipe.allow_up = self.scrollbar.has_next_page();
|
self.swipe.allow_up = self.scrollbar.has_next_page();
|
||||||
self.swipe.allow_down = self.scrollbar.has_previous_page();
|
self.swipe.allow_down = self.scrollbar.has_previous_page();
|
||||||
@ -76,20 +70,27 @@ where
|
|||||||
type Msg = PageMsg<T::Msg, U::Msg>;
|
type Msg = PageMsg<T::Msg, U::Msg>;
|
||||||
|
|
||||||
fn place(&mut self, bounds: Rect) -> Rect {
|
fn place(&mut self, bounds: Rect) -> Rect {
|
||||||
let layout = PageLayout::new(bounds, self.button_rows);
|
let layout = PageLayout::new(bounds);
|
||||||
self.pad.place(bounds);
|
self.pad.place(bounds);
|
||||||
self.swipe.place(bounds);
|
self.swipe.place(bounds);
|
||||||
self.hint.place(layout.hint);
|
self.hint.place(layout.hint);
|
||||||
self.buttons.place(layout.buttons);
|
let buttons_area = self.buttons.place(layout.buttons);
|
||||||
self.scrollbar.place(layout.scrollbar);
|
let buttons_height = buttons_area.height() + theme::BUTTON_SPACING;
|
||||||
|
self.scrollbar
|
||||||
|
.place(layout.scrollbar.inset(Insets::bottom(buttons_height)));
|
||||||
|
|
||||||
// Layout the content. Try to fit it on a single page first, and reduce the area
|
// Layout the content. Try to fit it on a single page first, and reduce the area
|
||||||
// to make space for a scrollbar if it doesn't fit.
|
// to make space for a scrollbar if it doesn't fit.
|
||||||
self.content.place(layout.content_single_page);
|
self.content.place(
|
||||||
|
layout
|
||||||
|
.content_single_page
|
||||||
|
.inset(Insets::bottom(buttons_height)),
|
||||||
|
);
|
||||||
let page_count = {
|
let page_count = {
|
||||||
let count = self.content.page_count();
|
let count = self.content.page_count();
|
||||||
if count > 1 {
|
if count > 1 {
|
||||||
self.content.place(layout.content);
|
self.content
|
||||||
|
.place(layout.content.inset(Insets::bottom(buttons_height)));
|
||||||
self.content.page_count() // Make sure to re-count it with the
|
self.content.page_count() // Make sure to re-count it with the
|
||||||
// new size.
|
// new size.
|
||||||
} else {
|
} else {
|
||||||
@ -197,14 +198,10 @@ impl PageLayout {
|
|||||||
const SCROLLBAR_SPACE: i32 = 10;
|
const SCROLLBAR_SPACE: i32 = 10;
|
||||||
const HINT_OFF: i32 = 19;
|
const HINT_OFF: i32 = 19;
|
||||||
|
|
||||||
pub fn new(area: Rect, button_rows: i32) -> Self {
|
pub fn new(area: Rect) -> Self {
|
||||||
let buttons_height = button_rows * Button::<&str>::HEIGHT
|
|
||||||
+ button_rows.saturating_sub(1) * theme::BUTTON_SPACING;
|
|
||||||
let (content, buttons) = area.split_bottom(buttons_height);
|
|
||||||
let (_, hint) = area.split_bottom(Self::HINT_OFF);
|
let (_, hint) = area.split_bottom(Self::HINT_OFF);
|
||||||
let (content, _space) = content.split_bottom(theme::BUTTON_SPACING);
|
let (buttons, _space) = area.split_right(theme::CONTENT_BORDER);
|
||||||
let (buttons, _space) = buttons.split_right(theme::CONTENT_BORDER);
|
let (_space, content) = area.split_left(theme::CONTENT_BORDER);
|
||||||
let (_space, content) = content.split_left(theme::CONTENT_BORDER);
|
|
||||||
let (content_single_page, _space) = content.split_right(theme::CONTENT_BORDER);
|
let (content_single_page, _space) = content.split_right(theme::CONTENT_BORDER);
|
||||||
let (content, scrollbar) =
|
let (content, scrollbar) =
|
||||||
content.split_right(Self::SCROLLBAR_SPACE + Self::SCROLLBAR_WIDTH);
|
content.split_right(Self::SCROLLBAR_SPACE + Self::SCROLLBAR_WIDTH);
|
||||||
@ -221,7 +218,7 @@ impl PageLayout {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct SwipeHoldPage<T> {
|
pub struct SwipeHoldPage<T> {
|
||||||
inner: SwipePage<T, CancelHold>,
|
inner: SwipePage<T, FixedHeightBar<CancelHold>>,
|
||||||
loader: Loader,
|
loader: Loader,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -328,7 +325,7 @@ mod tests {
|
|||||||
component::{text::paragraphs::Paragraphs, Empty},
|
component::{text::paragraphs::Paragraphs, Empty},
|
||||||
event::TouchEvent,
|
event::TouchEvent,
|
||||||
geometry::Point,
|
geometry::Point,
|
||||||
model_tt::{constant, theme},
|
model_tt::{component::Button, constant, theme},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -418,13 +415,13 @@ mod tests {
|
|||||||
theme::TEXT_BOLD,
|
theme::TEXT_BOLD,
|
||||||
"This is somewhat long paragraph that goes on and on and on and on and on and will definitely not fit on just a single screen. You have to swipe a bit to see all the text it contains I guess. There's just so much letters in it.",
|
"This is somewhat long paragraph that goes on and on and on and on and on and will definitely not fit on just a single screen. You have to swipe a bit to see all the text it contains I guess. There's just so much letters in it.",
|
||||||
),
|
),
|
||||||
Empty,
|
theme::button_bar(Button::with_text("NO")),
|
||||||
theme::BG,
|
theme::BG,
|
||||||
);
|
);
|
||||||
page.place(SCREEN);
|
page.place(SCREEN);
|
||||||
|
|
||||||
let expected1 = "<SwipePage active_page:0 page_count:2 content:<Paragraphs This is somewhat long\nparagraph that goes on\nand on and on and on\nand on and will definitely\nnot fit on just a single\nscreen. You have to\nswipe a bit to see all the\ntext it contains I guess....\n> buttons:<Empty > >";
|
let expected1 = "<SwipePage active_page:0 page_count:2 content:<Paragraphs This is somewhat long\nparagraph that goes on\nand on and on and on\nand on and will definitely\nnot fit on just a single\nscreen. You have to\nswipe a bit to see all the\ntext it contains I guess....\n> buttons:<FixedHeightBar inner:<Button text:NO > > >";
|
||||||
let expected2 = "<SwipePage active_page:1 page_count:2 content:<Paragraphs There's just so much\nletters in it.\n> buttons:<Empty > >";
|
let expected2 = "<SwipePage active_page:1 page_count:2 content:<Paragraphs There's just so much\nletters in it.\n> buttons:<FixedHeightBar inner:<Button text:NO > > >";
|
||||||
|
|
||||||
assert_eq!(trace(&page), expected1);
|
assert_eq!(trace(&page), expected1);
|
||||||
swipe_down(&mut page);
|
swipe_down(&mut page);
|
||||||
@ -453,14 +450,14 @@ mod tests {
|
|||||||
theme::TEXT_BOLD,
|
theme::TEXT_BOLD,
|
||||||
"Let's add another one for a good measure. This one should overflow all the way to the third page with a bit of luck.",
|
"Let's add another one for a good measure. This one should overflow all the way to the third page with a bit of luck.",
|
||||||
),
|
),
|
||||||
Empty,
|
theme::button_bar(Button::with_text("IDK")),
|
||||||
theme::BG,
|
theme::BG,
|
||||||
);
|
);
|
||||||
page.place(SCREEN);
|
page.place(SCREEN);
|
||||||
|
|
||||||
let expected1 = "<SwipePage active_page:0 page_count:3 content:<Paragraphs This paragraph is using a\nbold font. It doesn't\nneed to be all that long.\nAnd this one is\nusing MONO.\nMonospace is\nnice for...\n> buttons:<Empty > >";
|
let expected1 = "<SwipePage active_page:0 page_count:3 content:<Paragraphs This paragraph is using a\nbold font. It doesn't\nneed to be all that long.\nAnd this one is\nusing MONO.\nMonospace is\nnice for...\n> buttons:<FixedHeightBar inner:<Button text:IDK > > >";
|
||||||
let expected2 = "<SwipePage active_page:1 page_count:3 content:<Paragraphs numbers, they\nhave the same\nwidth and can be\nscanned quickly.\nEven if they\nspan several\npages or...\n> buttons:<Empty > >";
|
let expected2 = "<SwipePage active_page:1 page_count:3 content:<Paragraphs numbers, they\nhave the same\nwidth and can be\nscanned quickly.\nEven if they\nspan several\npages or...\n> buttons:<FixedHeightBar inner:<Button text:IDK > > >";
|
||||||
let expected3 = "<SwipePage active_page:2 page_count:3 content:<Paragraphs something.\nLet's add another one\nfor a good measure. This\none should overflow all\nthe way to the third\npage with a bit of luck.\n> buttons:<Empty > >";
|
let expected3 = "<SwipePage active_page:2 page_count:3 content:<Paragraphs something.\nLet's add another one\nfor a good measure. This\none should overflow all\nthe way to the third\npage with a bit of luck.\n> buttons:<FixedHeightBar inner:<Button text:IDK > > >";
|
||||||
|
|
||||||
assert_eq!(trace(&page), expected1);
|
assert_eq!(trace(&page), expected1);
|
||||||
swipe_down(&mut page);
|
swipe_down(&mut page);
|
||||||
@ -489,14 +486,14 @@ mod tests {
|
|||||||
.add_break()
|
.add_break()
|
||||||
.add(theme::TEXT_NORMAL, "Short three.")
|
.add(theme::TEXT_NORMAL, "Short three.")
|
||||||
.add_break(),
|
.add_break(),
|
||||||
Empty,
|
theme::button_bar(Empty),
|
||||||
theme::BG,
|
theme::BG,
|
||||||
);
|
);
|
||||||
page.place(SCREEN);
|
page.place(SCREEN);
|
||||||
|
|
||||||
let expected1 = "<SwipePage active_page:0 page_count:3 content:<Paragraphs Short one.\n> buttons:<Empty > >";
|
let expected1 = "<SwipePage active_page:0 page_count:3 content:<Paragraphs Short one.\n> buttons:<FixedHeightBar inner:<Empty > > >";
|
||||||
let expected2 = "<SwipePage active_page:1 page_count:3 content:<Paragraphs Short two.\n> buttons:<Empty > >";
|
let expected2 = "<SwipePage active_page:1 page_count:3 content:<Paragraphs Short two.\n> buttons:<FixedHeightBar inner:<Empty > > >";
|
||||||
let expected3 = "<SwipePage active_page:2 page_count:3 content:<Paragraphs Short three.\n> buttons:<Empty > >";
|
let expected3 = "<SwipePage active_page:2 page_count:3 content:<Paragraphs Short three.\n> buttons:<FixedHeightBar inner:<Empty > > >";
|
||||||
|
|
||||||
assert_eq!(trace(&page), expected1);
|
assert_eq!(trace(&page), expected1);
|
||||||
swipe_up(&mut page);
|
swipe_up(&mut page);
|
||||||
|
@ -586,9 +586,9 @@ extern "C" fn new_show_simple(n_args: usize, args: *const Obj, kwargs: *mut Map)
|
|||||||
t,
|
t,
|
||||||
Dialog::new(
|
Dialog::new(
|
||||||
Paragraphs::new().add(theme::TEXT_NORMAL, description),
|
Paragraphs::new().add(theme::TEXT_NORMAL, description),
|
||||||
Button::with_text(button).map(|msg| {
|
theme::button_bar(Button::with_text(button).map(|msg| {
|
||||||
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
||||||
}),
|
})),
|
||||||
),
|
),
|
||||||
))?
|
))?
|
||||||
.into()
|
.into()
|
||||||
@ -597,9 +597,9 @@ extern "C" fn new_show_simple(n_args: usize, args: *const Obj, kwargs: *mut Map)
|
|||||||
theme::borders(),
|
theme::borders(),
|
||||||
Dialog::new(
|
Dialog::new(
|
||||||
Paragraphs::new().add(theme::TEXT_NORMAL, description),
|
Paragraphs::new().add(theme::TEXT_NORMAL, description),
|
||||||
Button::with_text(button).map(|msg| {
|
theme::button_bar(Button::with_text(button).map(|msg| {
|
||||||
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
||||||
}),
|
})),
|
||||||
),
|
),
|
||||||
))?
|
))?
|
||||||
.into()
|
.into()
|
||||||
@ -629,11 +629,7 @@ extern "C" fn new_confirm_with_info(n_args: usize, args: *const Obj, kwargs: *mu
|
|||||||
let buttons = Button::cancel_info_confirm(button, info_button);
|
let buttons = Button::cancel_info_confirm(button, info_button);
|
||||||
|
|
||||||
let obj = LayoutObj::new(
|
let obj = LayoutObj::new(
|
||||||
Frame::new(
|
Frame::new(title, SwipePage::new(paragraphs, buttons, theme::BG)).into_child(),
|
||||||
title,
|
|
||||||
SwipePage::new(paragraphs, buttons, theme::BG).with_button_rows(2),
|
|
||||||
)
|
|
||||||
.into_child(),
|
|
||||||
)?;
|
)?;
|
||||||
Ok(obj.into())
|
Ok(obj.into())
|
||||||
};
|
};
|
||||||
@ -729,11 +725,7 @@ extern "C" fn new_select_word(n_args: usize, args: *const Obj, kwargs: *mut Map)
|
|||||||
let buttons = Button::select_word(words);
|
let buttons = Button::select_word(words);
|
||||||
|
|
||||||
let obj = LayoutObj::new(
|
let obj = LayoutObj::new(
|
||||||
Frame::new(
|
Frame::new(title, SwipePage::new(paragraphs, buttons, theme::BG)).into_child(),
|
||||||
title,
|
|
||||||
SwipePage::new(paragraphs, buttons, theme::BG).with_button_rows(3),
|
|
||||||
)
|
|
||||||
.into_child(),
|
|
||||||
)?;
|
)?;
|
||||||
Ok(obj.into())
|
Ok(obj.into())
|
||||||
};
|
};
|
||||||
@ -821,9 +813,9 @@ extern "C" fn new_show_checklist(n_args: usize, args: *const Obj, kwargs: *mut M
|
|||||||
active,
|
active,
|
||||||
paragraphs,
|
paragraphs,
|
||||||
),
|
),
|
||||||
Button::with_text(button).map(|msg| {
|
theme::button_bar(Button::with_text(button).map(|msg| {
|
||||||
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
(matches!(msg, ButtonMsg::Clicked)).then(|| CancelConfirmMsg::Confirmed)
|
||||||
}),
|
})),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
.with_border(theme::borders())
|
.with_border(theme::borders())
|
||||||
@ -1115,7 +1107,7 @@ mod tests {
|
|||||||
layout.place(SCREEN);
|
layout.place(SCREEN);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
trace(&layout),
|
trace(&layout),
|
||||||
"<Dialog content:<Text content:Testing text layout, with\nsome text, and some more\ntext. And parameters! > controls:<Tuple 0:<GridPlaced inner:<Button text:Left > > 1:<GridPlaced inner:<Button text:Right > > > >",
|
"<Dialog content:<Text content:Testing text layout, with\nsome text, and some more\ntext. And parameters! > controls:<FixedHeightBar inner:<Tuple 0:<GridPlaced inner:<Button text:Left > > 1:<GridPlaced inner:<Button text:Right > > > > >",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ use crate::ui::{
|
|||||||
component::{
|
component::{
|
||||||
label::LabelStyle,
|
label::LabelStyle,
|
||||||
text::{formatted::FormattedFonts, TextStyle},
|
text::{formatted::FormattedFonts, TextStyle},
|
||||||
|
FixedHeightBar,
|
||||||
},
|
},
|
||||||
display::{Color, Font},
|
display::{Color, Font},
|
||||||
geometry::Insets,
|
geometry::Insets,
|
||||||
@ -407,8 +408,22 @@ pub const FORMATTED: FormattedFonts = FormattedFonts {
|
|||||||
|
|
||||||
pub const CONTENT_BORDER: i32 = 5;
|
pub const CONTENT_BORDER: i32 = 5;
|
||||||
pub const KEYBOARD_SPACING: i32 = 8;
|
pub const KEYBOARD_SPACING: i32 = 8;
|
||||||
|
pub const BUTTON_HEIGHT: i32 = 38;
|
||||||
pub const BUTTON_SPACING: i32 = 6;
|
pub const BUTTON_SPACING: i32 = 6;
|
||||||
pub const CHECKLIST_SPACING: i32 = 10;
|
pub const CHECKLIST_SPACING: i32 = 10;
|
||||||
|
/// Standard button height in pixels.
|
||||||
|
pub const fn button_rows(count: usize) -> i32 {
|
||||||
|
let count = count as i32;
|
||||||
|
BUTTON_HEIGHT * count + BUTTON_SPACING * count.saturating_sub(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn button_bar_rows<T>(rows: usize, inner: T) -> FixedHeightBar<T> {
|
||||||
|
FixedHeightBar::bottom(inner, button_rows(rows))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn button_bar<T>(inner: T) -> FixedHeightBar<T> {
|
||||||
|
button_bar_rows(1, inner)
|
||||||
|
}
|
||||||
|
|
||||||
/// +----------+
|
/// +----------+
|
||||||
/// | 13 |
|
/// | 13 |
|
||||||
|
Loading…
Reference in New Issue
Block a user