1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-05 21:10:57 +00:00

feat(core/ui): add page counter to paginated blobs

This commit is contained in:
Ioan Bizău 2024-11-13 17:53:51 +01:00 committed by Martin Milata
parent ebc302959b
commit 97c9f84f8d
26 changed files with 103 additions and 56 deletions

View File

@ -0,0 +1 @@
[T3T1] Add page counter to paginated blobs.

View File

@ -357,6 +357,7 @@ static void _librust_qstrs(void) {
MP_QSTR_notification;
MP_QSTR_notification_level;
MP_QSTR_page_count;
MP_QSTR_page_counter;
MP_QSTR_page_limit;
MP_QSTR_pages;
MP_QSTR_paint;

View File

@ -150,7 +150,7 @@ where
}
impl<T: Paginate> Paginate for Child<T> {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.component.page_count()
}

View File

@ -22,7 +22,7 @@ pub enum PageMsg<T> {
pub trait Paginate {
/// How many pages of content are there in total?
fn page_count(&mut self) -> usize;
fn page_count(&self) -> usize;
/// Navigate to the given page.
fn change_page(&mut self, active_page: usize);
}

View File

@ -33,8 +33,17 @@ impl FormattedText {
}
pub(crate) fn layout_content(&self, sink: &mut dyn LayoutSink) -> LayoutFit {
self.layout_content_with_offset(sink, self.char_offset, self.y_offset)
}
fn layout_content_with_offset(
&self,
sink: &mut dyn LayoutSink,
char_offset: usize,
y_offset: i16,
) -> LayoutFit {
self.op_layout
.layout_ops(self.char_offset, Offset::y(self.y_offset), sink)
.layout_ops(char_offset, Offset::y(y_offset), sink)
}
fn align_vertically(&mut self, content_height: i16) {
@ -53,20 +62,15 @@ impl FormattedText {
// Pagination
impl Paginate for FormattedText {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
let mut page_count = 1; // There's always at least one page.
// Make sure we're starting page counting from the very beginning
// (but remembering the offsets not to change them).
let initial_y_offset = self.y_offset;
let initial_char_offset = self.char_offset;
self.char_offset = 0;
self.y_offset = 0;
let mut char_offset = 0;
// Looping through the content and counting pages
// until we finally fit.
loop {
let fit = self.layout_content(&mut TextNoOp);
let fit = self.layout_content_with_offset(&mut TextNoOp, char_offset, 0);
match fit {
LayoutFit::Fitting { .. } => {
break;
@ -75,15 +79,11 @@ impl Paginate for FormattedText {
processed_chars, ..
} => {
page_count += 1;
self.char_offset += processed_chars;
char_offset += processed_chars;
}
}
}
// Setting the offsets back to the initial values.
self.char_offset = initial_char_offset;
self.y_offset = initial_y_offset;
page_count
}

View File

@ -210,7 +210,7 @@ impl<'a, T> Paginate for Paragraphs<T>
where
T: ParagraphSource<'a>,
{
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
// There's always at least one page.
self.break_pages().count().max(1)
}
@ -374,7 +374,7 @@ struct PageOffset {
}
impl PageOffset {
/// Given an `PageOffset` and a `Rect` area, returns:
/// Given a `PageOffset` and a `Rect` area, returns:
///
/// - The next offset.
/// - Part of `area` that remains free after the current offset is rendered
@ -690,7 +690,7 @@ impl<'a, T> Paginate for Checklist<T>
where
T: ParagraphSource<'a>,
{
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
1
}

View File

@ -47,6 +47,14 @@ impl<T: Component + Paginate> SwipePage<T> {
self.limit = limit;
self
}
pub fn page_count(&self) -> usize {
self.pages
}
pub fn current_page(&self) -> usize {
self.current
}
}
impl<T: Component + Paginate> Component for SwipePage<T> {

View File

@ -117,7 +117,7 @@ impl AddressDetails {
}
impl Paginate for AddressDetails {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.get_internal_page_count()
}

View File

@ -69,11 +69,8 @@ impl<'a> Footer<'a> {
)
}
pub fn with_page_counter(max_pages: u8, instruction: TString<'static>) -> Self {
Self::from_content(FooterContent::PageCounter(PageCounter::new(
max_pages,
instruction,
)))
pub fn with_page_counter(instruction: TString<'static>) -> Self {
Self::from_content(FooterContent::PageCounter(PageCounter::new(instruction)))
}
pub fn with_page_hint(
@ -115,18 +112,18 @@ impl<'a> Footer<'a> {
}
}
pub fn update_page_counter(&mut self, ctx: &mut EventCtx, current: usize, max: Option<usize>) {
pub fn update_page_counter(&mut self, ctx: &mut EventCtx, current: usize, max: usize) {
match &mut self.content {
FooterContent::PageCounter(counter) => {
counter.update_current_page(current);
counter.update_current_page(current, max);
self.swipe_allow_down = counter.is_first_page();
self.swipe_allow_up = counter.is_last_page();
ctx.request_paint();
}
FooterContent::PageHint(page_hint) => {
page_hint.update_current_page(current, max);
self.swipe_allow_down = page_hint.is_first_page();
self.swipe_allow_up = page_hint.is_last_page();
FooterContent::PageHint(hint) => {
hint.update_current_page(current, max);
self.swipe_allow_down = hint.is_first_page();
self.swipe_allow_up = hint.is_last_page();
ctx.request_paint();
}
_ => {
@ -322,16 +319,17 @@ struct PageCounter {
}
impl PageCounter {
fn new(page_max: u8, instruction: TString<'static>) -> Self {
fn new(instruction: TString<'static>) -> Self {
Self {
instruction,
page_curr: 0,
page_max,
page_max: 0,
font: Font::SUB,
}
}
fn update_current_page(&mut self, new_value: usize) {
fn update_current_page(&mut self, new_value: usize, max: usize) {
self.page_max = max as u8;
self.page_curr = (new_value as u8).clamp(0, self.page_max.saturating_sub(1));
}
@ -410,11 +408,9 @@ struct PageHint {
}
impl PageHint {
fn update_current_page(&mut self, current: usize, max: Option<usize>) {
fn update_current_page(&mut self, current: usize, max: usize) {
self.page_num = max as u8;
self.page_curr = (current as u8).clamp(0, self.page_num.saturating_sub(1));
if let Some(max) = max {
self.page_num = max as u8;
}
}
fn update_max_page(&mut self, max: usize) {

View File

@ -195,8 +195,8 @@ where
}
#[inline(never)]
pub fn with_footer_counter(mut self, instruction: TString<'static>, max_value: u8) -> Self {
self.footer = Some(Footer::with_page_counter(max_value, instruction));
pub fn with_footer_counter(mut self, instruction: TString<'static>) -> Self {
self.footer = Some(Footer::with_page_counter(instruction));
self
}

View File

@ -335,7 +335,7 @@ impl<F: Fn(usize) -> TString<'static>> PagedVerticalMenu<F> {
}
impl<F: Fn(usize) -> TString<'static>> Paginate for PagedVerticalMenu<F> {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.num_pages()
}

View File

@ -7,7 +7,7 @@ use crate::{
component::{
swipe_detect::SwipeSettings,
text::paragraphs::{Paragraph, ParagraphSource, ParagraphVecShort, VecExt},
Component, ComponentExt, Paginate,
Component, ComponentExt, EventCtx, Paginate,
},
flow::{
base::{Decision, DecisionBuilder as _},
@ -19,7 +19,8 @@ use crate::{
use super::super::{
component::{
Frame, FrameMsg, PromptMsg, PromptScreen, SwipeContent, VerticalMenu, VerticalMenuChoiceMsg,
Footer, Frame, FrameMsg, PromptMsg, PromptScreen, SwipeContent, VerticalMenu,
VerticalMenuChoiceMsg,
},
theme,
};
@ -171,15 +172,17 @@ pub fn new_confirm_action(
ConfirmActionStrings::new(title, subtitle, None, prompt_screen.then_some(prompt_title)),
hold,
None,
false,
)
}
#[inline(never)]
fn new_confirm_action_uni<T: Component + MaybeTrace + 'static>(
content: T,
fn new_confirm_action_uni<T: Component + Paginate + MaybeTrace + 'static>(
content: SwipeContent<SwipePage<T>>,
menu: ConfirmActionMenu,
strings: ConfirmActionStrings,
hold: bool,
page_counter: bool,
) -> Result<SwipeFlow, error::Error> {
let (prompt_screen, prompt_pages, flow, page) =
create_flow(strings.title, strings.prompt_screen, hold);
@ -191,6 +194,22 @@ fn new_confirm_action_uni<T: Component + MaybeTrace + 'static>(
.with_menu_button()
.with_footer(TR::instructions__swipe_up.into(), None);
if page_counter {
fn footer_update_fn<T: Component + Paginate>(
content: &SwipeContent<SwipePage<T>>,
ctx: &mut EventCtx,
footer: &mut Footer,
) {
let current_page = content.inner().current_page();
let page_count = content.inner().page_count();
footer.update_page_counter(ctx, current_page, page_count);
}
content_intro = content_intro
.with_footer_counter(TR::instructions__swipe_up.into())
.register_footer_update_fn(footer_update_fn::<T>);
}
if let Some(subtitle) = strings.subtitle {
content_intro = content_intro.with_subtitle(subtitle);
}
@ -318,11 +337,13 @@ pub fn new_confirm_action_simple<T: Component + Paginate + MaybeTrace + 'static>
strings: ConfirmActionStrings,
hold: bool,
page_limit: Option<usize>,
page_counter: bool,
) -> Result<SwipeFlow, error::Error> {
new_confirm_action_uni(
SwipeContent::new(SwipePage::vertical(content).with_limit(page_limit)),
menu,
strings,
hold,
page_counter,
)
}

View File

@ -100,7 +100,7 @@ fn footer_update_fn(
) {
let current_page = content.inner().inner().current_page();
let total_pages = content.inner().inner().num_pages();
footer.update_page_counter(ctx, current_page, Some(total_pages));
footer.update_page_counter(ctx, current_page, total_pages);
}
impl ConfirmFido {

View File

@ -151,7 +151,7 @@ fn footer_update_fn(
// to get total pages instead of using Paginate because it borrows mutably
let current_page = content.inner().inner().current_page();
let total_pages = content.inner().inner().inner().len() / 2; // 2 paragraphs per page
footer.update_page_counter(ctx, current_page, Some(total_pages));
footer.update_page_counter(ctx, current_page, total_pages);
}
pub fn new_continue_recovery(

View File

@ -74,7 +74,7 @@ fn footer_updating_func(
) {
let current_page = content.inner().current_page();
let total_pages = content.inner().num_pages();
footer.update_page_counter(ctx, current_page, Some(total_pages));
footer.update_page_counter(ctx, current_page, total_pages);
}
pub fn new_show_share_words(
@ -103,7 +103,6 @@ pub fn new_show_share_words(
.one_button_request(ButtonRequestCode::ResetDevice.with_name("share_words"))
.with_pages(move |_| nwords + 2);
let n_words = share_words_vec.len();
let content_words = Frame::left_aligned(
title,
InternallySwipableContent::new(ShareWords::new(share_words_vec, subtitle)),
@ -113,7 +112,7 @@ pub fn new_show_share_words(
.with_vertical_pages()
.with_subtitle(subtitle)
.register_header_update_fn(header_updating_func)
.with_footer_counter(TR::instructions__swipe_up.into(), n_words as u8)
.with_footer_counter(TR::instructions__swipe_up.into())
.register_footer_update_fn(footer_updating_func)
.map(|_| None);

View File

@ -47,6 +47,7 @@ pub struct ConfirmBlobParams {
hold: bool,
chunkify: bool,
text_mono: bool,
page_counter: bool,
page_limit: Option<usize>,
swipe_up: bool,
swipe_down: bool,
@ -78,6 +79,7 @@ impl ConfirmBlobParams {
hold: false,
chunkify: false,
text_mono: true,
page_counter: false,
page_limit: None,
swipe_up: false,
swipe_down: false,
@ -170,6 +172,11 @@ impl ConfirmBlobParams {
self
}
pub fn with_page_counter(mut self, page_counter: bool) -> Self {
self.page_counter = page_counter;
self
}
pub const fn with_page_limit(mut self, page_limit: Option<usize>) -> Self {
self.page_limit = page_limit;
self
@ -266,6 +273,7 @@ impl ConfirmBlobParams {
),
self.hold,
self.page_limit,
self.page_counter,
)
}
}

View File

@ -250,6 +250,7 @@ extern "C" fn new_confirm_emphasized(n_args: usize, args: *const Obj, kwargs: *m
ConfirmActionStrings::new(title, None, None, Some(title)),
false,
None,
false,
)
.and_then(LayoutObj::new_root)
.map(Into::into)
@ -289,6 +290,7 @@ extern "C" fn new_confirm_blob(n_args: usize, args: *const Obj, kwargs: *mut Map
let info: bool = kwargs.get_or(Qstr::MP_QSTR_info, true)?;
let hold: bool = kwargs.get_or(Qstr::MP_QSTR_hold, false)?;
let chunkify: bool = kwargs.get_or(Qstr::MP_QSTR_chunkify, false)?;
let page_counter: bool = kwargs.get_or(Qstr::MP_QSTR_page_counter, false)?;
let prompt_screen: bool = kwargs.get_or(Qstr::MP_QSTR_prompt_screen, true)?;
let page_limit: Option<usize> = kwargs
.get(Qstr::MP_QSTR_page_limit)
@ -314,6 +316,7 @@ extern "C" fn new_confirm_blob(n_args: usize, args: *const Obj, kwargs: *mut Map
.with_extra(extra)
.with_info_button(info)
.with_chunkify(chunkify)
.with_page_counter(page_counter)
.with_page_limit(page_limit)
.with_prompt(prompt_screen)
.with_hold(hold)
@ -427,6 +430,7 @@ extern "C" fn new_confirm_properties(n_args: usize, args: *const Obj, kwargs: *m
ConfirmActionStrings::new(title, None, None, hold.then_some(title)),
hold,
None,
false,
)
.and_then(LayoutObj::new_root)
.map(Into::into)
@ -461,6 +465,7 @@ extern "C" fn new_confirm_homescreen(n_args: usize, args: *const Obj, kwargs: *m
),
false,
None,
false,
)
.and_then(LayoutObj::new_root)
.map(Into::into)
@ -767,6 +772,7 @@ extern "C" fn new_confirm_total(n_args: usize, args: *const Obj, kwargs: *mut Ma
ConfirmActionStrings::new(title, None, None, Some(title)),
true,
None,
false,
)
.and_then(LayoutObj::new_root)
.map(Into::into)
@ -1087,6 +1093,7 @@ extern "C" fn new_confirm_coinjoin(n_args: usize, args: *const Obj, kwargs: *mut
),
true,
None,
false,
)
.and_then(LayoutObj::new_root)
.map(Into::into)
@ -1570,6 +1577,7 @@ pub static mp_module_trezorui2: Module = obj_module! {
/// info: bool = True,
/// hold: bool = False,
/// chunkify: bool = False,
/// page_counter: bool = False,
/// prompt_screen: bool = False,
/// page_limit: int | None = None,
/// ) -> LayoutObj[UiResult]:

View File

@ -199,7 +199,7 @@ impl Page {
// Pagination
impl Paginate for Page {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.formatted.page_count()
}

View File

@ -87,7 +87,7 @@ impl<T> Paginate for Frame<T>
where
T: Component + Paginate,
{
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.content.page_count()
}

View File

@ -225,7 +225,7 @@ impl Component for ScrollBar {
}
impl Paginate for ScrollBar {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
self.page_count
}

View File

@ -150,8 +150,7 @@ impl<'a> Component for ShareWords<'a> {
}
impl<'a> Paginate for ShareWords<'a> {
fn page_count(&mut self) -> usize {
// Not defining the logic here, as we do not want it to be `&mut`.
fn page_count(&self) -> usize {
self.total_page_count()
}

View File

@ -1709,6 +1709,7 @@ pub static mp_module_trezorui2: Module = obj_module! {
/// info: bool = True,
/// hold: bool = False,
/// chunkify: bool = False,
/// page_counter: bool = False,
/// prompt_screen: bool = False,
/// page_limit: int | None = None,
/// ) -> LayoutObj[UiResult]:

View File

@ -125,7 +125,7 @@ impl AddressDetails {
}
impl Paginate for AddressDetails {
fn page_count(&mut self) -> usize {
fn page_count(&self) -> usize {
let total_xpub_pages: u8 = self.xpub_page_count.iter().copied().sum();
2usize.saturating_add(total_xpub_pages.into())
}

View File

@ -1803,6 +1803,7 @@ pub static mp_module_trezorui2: Module = obj_module! {
/// info: bool = True,
/// hold: bool = False,
/// chunkify: bool = False,
/// page_counter: bool = False,
/// prompt_screen: bool = False,
/// page_limit: int | None = None,
/// ) -> LayoutObj[UiResult]:

View File

@ -68,6 +68,7 @@ def confirm_blob(
info: bool = True,
hold: bool = False,
chunkify: bool = False,
page_counter: bool = False,
prompt_screen: bool = False,
page_limit: int | None = None,
) -> LayoutObj[UiResult]:
@ -640,6 +641,7 @@ def confirm_blob(
info: bool = True,
hold: bool = False,
chunkify: bool = False,
page_counter: bool = False,
prompt_screen: bool = False,
page_limit: int | None = None,
) -> LayoutObj[UiResult]:
@ -1211,6 +1213,7 @@ def confirm_blob(
info: bool = True,
hold: bool = False,
chunkify: bool = False,
page_counter: bool = False,
prompt_screen: bool = False,
page_limit: int | None = None,
) -> LayoutObj[UiResult]:

View File

@ -486,6 +486,7 @@ def confirm_blob(
info=False,
hold=False,
chunkify=chunkify,
page_counter=True,
prompt_screen=False,
)