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

refactor(core/ui): improve LayoutObj initialization

* RequestPaint message is sent at construction time to force calculation
  of number of pages
* given that Attach corresponds to "start the layout" message, Child now
  responds to Attach the same way it responds to RequestPaint, by
  force-repainting everything.
This commit is contained in:
matejcik 2023-11-06 11:42:24 +01:00 committed by M1nd3r
parent 027a6ce8f0
commit e7bc73bebc
2 changed files with 34 additions and 31 deletions

View File

@ -142,7 +142,7 @@ where
// Handle the internal invalidation event here, so components don't have to. We // Handle the internal invalidation event here, so components don't have to. We
// still pass it inside, so the event propagates correctly to all components in // still pass it inside, so the event propagates correctly to all components in
// the sub-tree. // the sub-tree.
if let Event::RequestPaint = event { if matches!(event, Event::RequestPaint | Event::Attach(_)) {
ctx.request_paint(); ctx.request_paint();
} }
c.event(ctx, event) c.event(ctx, event)
@ -370,8 +370,9 @@ pub enum Event {
Timer(TimerToken), Timer(TimerToken),
/// Advance progress bar. Progress screens only. /// Advance progress bar. Progress screens only.
Progress(u16, TString<'static>), Progress(u16, TString<'static>),
/// Component has been attached to component tree. This event is sent once /// Component has been attached to component tree, all children should
/// before any other events. /// prepare for painting and/or start their timers.
/// This event is sent once before any other events.
Attach(AttachType), Attach(AttachType),
/// Internally-handled event to inform all `Child` wrappers in a sub-tree to /// Internally-handled event to inform all `Child` wrappers in a sub-tree to
/// get scheduled for painting. /// get scheduled for painting.
@ -443,9 +444,8 @@ impl EventCtx {
Self { Self {
timers: Vec::new(), timers: Vec::new(),
next_token: Self::STARTING_TIMER_TOKEN, next_token: Self::STARTING_TIMER_TOKEN,
place_requested: true, // We need to perform a place pass in the beginning. place_requested: false,
paint_requested: false, /* We also need to paint, but this is supplemented by paint_requested: false,
* `Child::marked_for_paint` being true. */
anim_frame_scheduled: false, anim_frame_scheduled: false,
page_count: None, page_count: None,
button_request: None, button_request: None,
@ -548,17 +548,13 @@ impl EventCtx {
} }
pub fn clear(&mut self) { pub fn clear(&mut self) {
self.place_requested = false;
self.paint_requested = false;
self.anim_frame_scheduled = false;
self.page_count = None;
#[cfg(feature = "ui_debug")] #[cfg(feature = "ui_debug")]
assert!(self.button_request.is_none()); assert!(self.button_request.is_none());
self.button_request = None; // replace self with a new instance, keeping only the fields we care about
self.root_repaint_requested = false; *self = Self {
self.swipe_disable_req = false; next_token: self.next_token,
self.swipe_enable_req = false; ..Self::new()
self.transition_out = None; }
} }
fn register_timer(&mut self, token: TimerToken, duration: Duration) { fn register_timer(&mut self, token: TimerToken, duration: Duration) {

View File

@ -128,6 +128,7 @@ where
} }
#[derive(Copy, Clone, PartialEq, Eq)] #[derive(Copy, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "debug", derive(ufmt::derive::uDebug))]
enum Repaint { enum Repaint {
None, None,
Partial, Partial,
@ -158,14 +159,22 @@ impl LayoutObjInner {
pub fn new(root: impl ObjComponent + 'static) -> Result<Self, Error> { pub fn new(root: impl ObjComponent + 'static) -> Result<Self, Error> {
let root = GcBox::new(root)?; let root = GcBox::new(root)?;
Ok(Self { let mut new = Self {
root: Some(gc::coerce!(ObjComponent, root)), root: Some(gc::coerce!(ObjComponent, root)),
event_ctx: EventCtx::new(), event_ctx: EventCtx::new(),
timer_fn: Obj::const_none(), timer_fn: Obj::const_none(),
page_count: 1, page_count: 1,
repaint: Repaint::Full, repaint: Repaint::Full,
transition_out: AttachType::Initial, transition_out: AttachType::Initial,
}) };
// invoke the initial placement
new.root_mut().obj_place(constant::screen());
// cause a repaint pass to update the number of pages
let msg = new.obj_event(Event::RequestPaint);
assert!(matches!(msg, Ok(s) if s == Obj::const_none()));
Ok(new)
} }
fn obj_delete(&mut self) { fn obj_delete(&mut self) {
@ -188,14 +197,14 @@ impl LayoutObjInner {
fn obj_request_repaint(&mut self) { fn obj_request_repaint(&mut self) {
self.repaint = Repaint::Full; self.repaint = Repaint::Full;
let mut dummy_ctx = EventCtx::new(); let mut event_ctx = EventCtx::new();
let paint_msg = self let paint_msg = self
.root_mut() .root_mut()
.obj_event(&mut dummy_ctx, Event::RequestPaint); .obj_event(&mut event_ctx, Event::RequestPaint);
// paint_msg must not be an error and it must not return a result // paint_msg must not be an error and it must not return a result
assert!(matches!(paint_msg, Ok(s) if s == Obj::const_none())); assert!(matches!(paint_msg, Ok(s) if s == Obj::const_none()));
// there must be no timers set // there must be no timers set
assert!(dummy_ctx.pop_timer().is_none()); assert!(event_ctx.pop_timer().is_none());
} }
/// Run an event pass over the component tree. After the traversal, any /// Run an event pass over the component tree. After the traversal, any
@ -204,21 +213,22 @@ impl LayoutObjInner {
/// an error, `Ok` with the message otherwise. /// an error, `Ok` with the message otherwise.
fn obj_event(&mut self, event: Event) -> Result<Obj, Error> { fn obj_event(&mut self, event: Event) -> Result<Obj, Error> {
let root = unwrap!(self.root.as_mut()); let root = unwrap!(self.root.as_mut());
// Place the root component on the screen in case it was previously requested.
if self.event_ctx.needs_place() {
root.obj_place(constant::screen());
}
// Clear the leftover flags from the previous event pass. // Get the event context ready for a new event
self.event_ctx.clear(); self.event_ctx.clear();
// Send the event down the component tree. Bail out in case of failure. // Send the event down the component tree. Bail out in case of failure.
let msg = root.obj_event(&mut self.event_ctx, event)?; let msg = root.obj_event(&mut self.event_ctx, event)?;
// Place the root component on the screen in case it was requested.
if self.event_ctx.needs_place() {
root.obj_place(constant::screen());
}
// Check if we should repaint next time // Check if we should repaint next time
if self.event_ctx.needs_repaint_root() { if self.event_ctx.needs_repaint_root() {
self.obj_request_repaint(); self.obj_request_repaint();
} else if self.event_ctx.needs_repaint() { } else if self.event_ctx.needs_repaint() && self.repaint == Repaint::None {
self.repaint = Repaint::Partial; self.repaint = Repaint::Partial;
} }
@ -236,10 +246,12 @@ impl LayoutObjInner {
} }
} }
// Update page count if it changed
if let Some(count) = self.event_ctx.page_count() { if let Some(count) = self.event_ctx.page_count() {
self.page_count = count as u16; self.page_count = count as u16;
} }
// Update outgoing transition if set
if let Some(t) = self.event_ctx.get_transition_out() { if let Some(t) = self.event_ctx.get_transition_out() {
self.transition_out = t; self.transition_out = t;
} }
@ -254,11 +266,6 @@ impl LayoutObjInner {
display::clear(); display::clear();
} }
// Place the root component on the screen in case it was previously requested.
if self.event_ctx.needs_place() {
self.root_mut().obj_place(constant::screen());
}
display::sync(); display::sync();
if self.repaint != Repaint::None { if self.repaint != Repaint::None {