mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-12-22 14:28:07 +00:00
feat(core/rust/ui): add model T title bar
[no changelog]
This commit is contained in:
parent
c7b33e2bc0
commit
f167a2bef2
@ -5,13 +5,13 @@ use crate::ui::{
|
|||||||
geometry::{Offset, Rect},
|
geometry::{Offset, Rect},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct Title<T, U> {
|
pub struct Frame<T, U> {
|
||||||
area: Rect,
|
area: Rect,
|
||||||
title: U,
|
title: U,
|
||||||
content: Child<T>,
|
content: Child<T>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Component, U: AsRef<[u8]>> Title<T, U> {
|
impl<T: Component, U: AsRef<[u8]>> Frame<T, U> {
|
||||||
pub fn new(area: Rect, title: U, content: impl FnOnce(Rect) -> T) -> Self {
|
pub fn new(area: Rect, title: U, content: impl FnOnce(Rect) -> T) -> Self {
|
||||||
let (title_area, content_area) = Self::areas(area);
|
let (title_area, content_area) = Self::areas(area);
|
||||||
Self {
|
Self {
|
||||||
@ -32,7 +32,7 @@ impl<T: Component, U: AsRef<[u8]>> Title<T, U> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Component, U: AsRef<[u8]>> Component for Title<T, U> {
|
impl<T: Component, U: AsRef<[u8]>> Component for Frame<T, U> {
|
||||||
type Msg = T::Msg;
|
type Msg = T::Msg;
|
||||||
|
|
||||||
fn event(&mut self, ctx: &mut EventCtx, event: Event) -> Option<Self::Msg> {
|
fn event(&mut self, ctx: &mut EventCtx, event: Event) -> Option<Self::Msg> {
|
||||||
@ -53,13 +53,13 @@ impl<T: Component, U: AsRef<[u8]>> Component for Title<T, U> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "ui_debug")]
|
#[cfg(feature = "ui_debug")]
|
||||||
impl<T, U> crate::trace::Trace for Title<T, U>
|
impl<T, U> crate::trace::Trace for Frame<T, U>
|
||||||
where
|
where
|
||||||
T: crate::trace::Trace,
|
T: crate::trace::Trace,
|
||||||
U: crate::trace::Trace + AsRef<[u8]>,
|
U: crate::trace::Trace + AsRef<[u8]>,
|
||||||
{
|
{
|
||||||
fn trace(&self, t: &mut dyn crate::trace::Tracer) {
|
fn trace(&self, t: &mut dyn crate::trace::Tracer) {
|
||||||
t.open("Title");
|
t.open("Frame");
|
||||||
t.field("title", &self.title);
|
t.field("title", &self.title);
|
||||||
t.field("content", &self.content);
|
t.field("content", &self.content);
|
||||||
t.close();
|
t.close();
|
@ -1,11 +1,11 @@
|
|||||||
mod button;
|
mod button;
|
||||||
mod dialog;
|
mod dialog;
|
||||||
|
mod frame;
|
||||||
mod page;
|
mod page;
|
||||||
mod title;
|
|
||||||
|
|
||||||
use super::{event, theme};
|
use super::{event, theme};
|
||||||
|
|
||||||
pub use button::{Button, ButtonContent, ButtonMsg, ButtonPos, ButtonStyle, ButtonStyleSheet};
|
pub use button::{Button, ButtonContent, ButtonMsg, ButtonPos, ButtonStyle, ButtonStyleSheet};
|
||||||
pub use dialog::{Dialog, DialogMsg};
|
pub use dialog::{Dialog, DialogMsg};
|
||||||
|
pub use frame::Frame;
|
||||||
pub use page::ButtonPage;
|
pub use page::ButtonPage;
|
||||||
pub use title::Title;
|
|
||||||
|
@ -12,7 +12,7 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
component::{Button, ButtonPage, Title},
|
component::{Button, ButtonPage, Frame},
|
||||||
theme,
|
theme,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -56,7 +56,7 @@ extern "C" fn ui_layout_new_confirm_action(
|
|||||||
let right = verb
|
let right = verb
|
||||||
.map(|label| |area, pos| Button::with_text(area, pos, label, theme::button_default()));
|
.map(|label| |area, pos| Button::with_text(area, pos, label, theme::button_default()));
|
||||||
|
|
||||||
let obj = LayoutObj::new(Child::new(Title::new(display::screen(), title, |area| {
|
let obj = LayoutObj::new(Child::new(Frame::new(display::screen(), title, |area| {
|
||||||
ButtonPage::new(
|
ButtonPage::new(
|
||||||
area,
|
area,
|
||||||
|area| {
|
|area| {
|
||||||
@ -84,7 +84,7 @@ extern "C" fn ui_layout_new_confirm_text(
|
|||||||
let description: Option<Buffer> =
|
let description: Option<Buffer> =
|
||||||
kwargs.get(Qstr::MP_QSTR_description)?.try_into_option()?;
|
kwargs.get(Qstr::MP_QSTR_description)?.try_into_option()?;
|
||||||
|
|
||||||
let obj = LayoutObj::new(Child::new(Title::new(display::screen(), title, |area| {
|
let obj = LayoutObj::new(Child::new(Frame::new(display::screen(), title, |area| {
|
||||||
ButtonPage::new(
|
ButtonPage::new(
|
||||||
area,
|
area,
|
||||||
|area| {
|
|area| {
|
||||||
@ -194,7 +194,7 @@ arameters! > left:<Button text:Left > right:<Button text:Right > >"#
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn trace_layout_title() {
|
fn trace_layout_title() {
|
||||||
let layout = Child::new(Title::new(display::screen(), "Please confirm", |area| {
|
let layout = Child::new(Frame::new(display::screen(), "Please confirm", |area| {
|
||||||
Dialog::new(
|
Dialog::new(
|
||||||
area,
|
area,
|
||||||
|area| {
|
|area| {
|
||||||
@ -210,7 +210,7 @@ arameters! > left:<Button text:Left > right:<Button text:Right > >"#
|
|||||||
}));
|
}));
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
trace(&layout),
|
trace(&layout),
|
||||||
r#"<Title title:Please confirm content:<Dialog content:<Text content:Testing text layout,
|
r#"<Frame title:Please confirm content:<Dialog content:<Text content:Testing text layout,
|
||||||
with some text, and
|
with some text, and
|
||||||
some more text. And p-
|
some more text. And p-
|
||||||
arameters! > left:<Button text:Left > right:<Button text:Right > > >"#
|
arameters! > left:<Button text:Left > right:<Button text:Right > > >"#
|
||||||
|
80
core/embed/rust/src/ui/model_tt/component/frame.rs
Normal file
80
core/embed/rust/src/ui/model_tt/component/frame.rs
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
use super::theme;
|
||||||
|
use crate::ui::{
|
||||||
|
component::{Child, Component, ComponentExt, Event, EventCtx},
|
||||||
|
display,
|
||||||
|
geometry::Rect,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct Frame<T, U> {
|
||||||
|
area: Rect,
|
||||||
|
title: U,
|
||||||
|
content: Child<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, U> Frame<T, U>
|
||||||
|
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) {
|
||||||
|
const HEADER_SPACE: i32 = 14;
|
||||||
|
let header_height = theme::FONT_BOLD.line_height() - theme::CONTENT_BORDER;
|
||||||
|
|
||||||
|
let (header_area, content_area) = area.hsplit(header_height);
|
||||||
|
let (_space, header_area) = header_area.vsplit(theme::CONTENT_BORDER);
|
||||||
|
let (_space, content_area) = content_area.hsplit(HEADER_SPACE);
|
||||||
|
|
||||||
|
(header_area, content_area)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, U> Component for Frame<T, U>
|
||||||
|
where
|
||||||
|
T: Component,
|
||||||
|
U: AsRef<[u8]>,
|
||||||
|
{
|
||||||
|
type Msg = T::Msg;
|
||||||
|
|
||||||
|
fn event(&mut self, ctx: &mut EventCtx, event: Event) -> Option<Self::Msg> {
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "ui_debug")]
|
||||||
|
impl<T, U> crate::trace::Trace for Frame<T, U>
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn bounds(&self, sink: &dyn Fn(Rect)) {
|
||||||
|
sink(self.area);
|
||||||
|
self.content.bounds(sink);
|
||||||
|
}
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
mod button;
|
mod button;
|
||||||
mod confirm;
|
mod confirm;
|
||||||
mod dialog;
|
mod dialog;
|
||||||
|
mod frame;
|
||||||
mod loader;
|
mod loader;
|
||||||
mod page;
|
mod page;
|
||||||
mod passphrase;
|
mod passphrase;
|
||||||
@ -10,6 +11,7 @@ mod swipe;
|
|||||||
pub use button::{Button, ButtonContent, ButtonMsg, ButtonStyle, ButtonStyleSheet};
|
pub use button::{Button, ButtonContent, ButtonMsg, ButtonStyle, ButtonStyleSheet};
|
||||||
pub use confirm::{HoldToConfirm, HoldToConfirmMsg};
|
pub use confirm::{HoldToConfirm, HoldToConfirmMsg};
|
||||||
pub use dialog::{Dialog, DialogLayout, DialogMsg};
|
pub use dialog::{Dialog, DialogLayout, DialogMsg};
|
||||||
|
pub use frame::Frame;
|
||||||
pub use loader::{Loader, LoaderMsg, LoaderStyle, LoaderStyleSheet};
|
pub use loader::{Loader, LoaderMsg, LoaderStyle, LoaderStyleSheet};
|
||||||
pub use page::SwipePage;
|
pub use page::SwipePage;
|
||||||
pub use swipe::{Swipe, SwipeDirection};
|
pub use swipe::{Swipe, SwipeDirection};
|
||||||
|
Loading…
Reference in New Issue
Block a user