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:
parent
027a6ce8f0
commit
e7bc73bebc
@ -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) {
|
||||||
|
@ -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 {
|
||||||
|
Loading…
Reference in New Issue
Block a user