1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-03 12:00:59 +00:00

feat(core/rust): improve the painting of TR's buttons + new inverse design

[no changelog]
This commit is contained in:
grdddj 2023-06-16 15:40:00 +02:00 committed by Jiří Musil
parent a6316f964e
commit b420064c08
2 changed files with 85 additions and 114 deletions

View File

@ -5,7 +5,7 @@ use crate::{
component::{Component, Event, EventCtx, Never},
constant,
display::{self, Color, Font, Icon},
geometry::{Alignment2D, Insets, Offset, Point, Rect},
geometry::{Alignment2D, Offset, Point, Rect},
},
};
@ -115,49 +115,40 @@ where
let button_width = if let Some(width) = style.fixed_width {
width
} else {
let outline = if style.with_outline {
theme::BUTTON_OUTLINE
} else {
0
};
let content_width = match &self.content {
ButtonContent::Text(text) => style.font.visible_text_width(text.as_ref()),
ButtonContent::Icon(icon) => icon.toif.width(),
};
content_width + 2 * outline
};
// Button height may be adjusted for the icon without outline
// Done to avoid highlighting bigger area than necessary when
// drawing the icon in active (black on white) state
let button_height = match &self.content {
ButtonContent::Text(_) => theme::BUTTON_HEIGHT,
ButtonContent::Icon(icon) => {
if style.with_outline {
theme::BUTTON_HEIGHT
} else {
icon.toif.height()
match &self.content {
ButtonContent::Text(text) => {
let text_width = style.font.visible_text_width(text.as_ref());
if style.with_outline {
text_width + 2 * theme::BUTTON_OUTLINE
} else if style.with_arms {
text_width + 2 * theme::ARMS_MARGIN
} else {
text_width
}
}
ButtonContent::Icon(icon) => {
// When Icon does not have outline, hardcode its width
if style.with_outline {
icon.toif.width() + 2 * theme::BUTTON_OUTLINE
} else {
theme::BUTTON_ICON_WIDTH
}
}
}
};
let button_bounds = self.bounds.split_bottom(button_height).1;
let area = match self.pos {
let button_bounds = self.bounds.split_bottom(theme::BUTTON_HEIGHT).1;
match self.pos {
ButtonPos::Left => button_bounds.split_left(button_width).0,
ButtonPos::Right => button_bounds.split_right(button_width).1,
ButtonPos::Middle => button_bounds.split_center(button_width).1,
};
// Allowing for possible offset of the area from current style
area.translate(style.offset)
}
}
/// Determine baseline point for the text.
fn get_text_baseline(&self, style: &ButtonStyle) -> Point {
// Arms and outline require the text to be elevated.
let offset_y = if style.with_arms {
theme::BUTTON_ARMS
} else if style.with_outline {
let offset_y = if style.with_outline || style.with_arms {
theme::BUTTON_OUTLINE
} else {
0
@ -165,6 +156,8 @@ where
let offset_x = if style.with_outline {
theme::BUTTON_OUTLINE
} else if style.with_arms {
theme::ARMS_MARGIN
} else {
0
};
@ -191,51 +184,41 @@ where
fn paint(&mut self) {
let style = self.style();
let text_color = style.text_color;
let background_color = text_color.negate();
let mut area = self.get_current_area();
let fg_color = style.text_color;
let bg_color = fg_color.negate();
let area = self.get_current_area();
let inversed_colors = bg_color != theme::BG;
// Optionally display "arms" at both sides of content, or create
// a nice rounded outline around it.
// By default just fill the content background.
if style.with_arms {
const ARM_WIDTH: i16 = 15;
area = area.translate(Offset::y(1));
// Prepare space for both the arms and content with BG color.
// Arms are icons 10*6 pixels.
let area_to_fill = area.outset(Insets::sides(ARM_WIDTH));
display::rect_fill(area_to_fill, background_color);
display::rect_fill_corners(area_to_fill, theme::BG);
// Paint both arms.
// Baselines are adjusted to give space between text and icon.
// 2 px because 1px might lead to odd coordinate which can't be render
theme::ICON_ARM_LEFT.draw(
area.left_center() - Offset::x(2),
Alignment2D::TOP_RIGHT,
text_color,
background_color,
);
theme::ICON_ARM_RIGHT.draw(
area.right_center() + Offset::x(2),
Alignment2D::TOP_LEFT,
text_color,
background_color,
);
// Filling the background (with 2-pixel rounding when applicable)
if inversed_colors {
display::rect_outline_rounded(area, bg_color, fg_color, 2);
display::rect_fill(area.shrink(1), bg_color);
} else if style.with_outline {
if background_color == theme::BG {
display::rect_outline_rounded(area, text_color, background_color, 2);
} else {
// With inverse colors having just radius of one, `rect_outline_rounded`
// is not suitable for inverse colors.
display::rect_fill(area, background_color);
display::rect_fill_corners(area, theme::BG);
}
display::rect_outline_rounded(area, fg_color, bg_color, 2);
} else {
display::rect_fill(area, background_color);
display::rect_fill(area, bg_color);
}
// Optionally display "arms" at both sides of content - always in FG and BG
// colors (they are not inverted).
if style.with_arms {
// Putting them one pixel down to touch the bottom.
let arms_area = area.translate(Offset::y(1));
theme::ICON_ARM_LEFT.draw(
arms_area.left_center(),
Alignment2D::TOP_RIGHT,
theme::FG,
theme::BG,
);
theme::ICON_ARM_RIGHT.draw(
arms_area.right_center(),
Alignment2D::TOP_LEFT,
theme::FG,
theme::BG,
);
}
// Painting the content
match &self.content {
ButtonContent::Text(text) => {
display::text_left(
@ -243,37 +226,33 @@ where
- Offset::x(style.font.start_x_bearing(text.as_ref())),
text.as_ref(),
style.font,
text_color,
background_color,
fg_color,
bg_color,
);
}
ButtonContent::Icon(icon) => {
// Allowing for possible offset of the area from current style
let icon_area = area.translate(style.offset);
if style.with_outline {
// Accounting for the 8*8 icon with empty left column and bottom row
// (which fits the outline nicely and symmetrically)
let center = area.center() + Offset::uniform(1);
icon.draw(center, Alignment2D::CENTER, text_color, background_color);
icon.draw(icon_area.center(), Alignment2D::CENTER, fg_color, bg_color);
} else {
// Positioning the icon in the corresponding corner/center
match self.pos {
ButtonPos::Left => icon.draw(
area.bottom_left(),
icon_area.bottom_left(),
Alignment2D::BOTTOM_LEFT,
text_color,
background_color,
fg_color,
bg_color,
),
ButtonPos::Right => icon.draw(
area.bottom_right(),
icon_area.bottom_right(),
Alignment2D::BOTTOM_RIGHT,
text_color,
background_color,
),
ButtonPos::Middle => icon.draw(
area.center(),
Alignment2D::CENTER,
text_color,
background_color,
fg_color,
bg_color,
),
ButtonPos::Middle => {
icon.draw(icon_area.center(), Alignment2D::CENTER, fg_color, bg_color)
}
}
}
}
@ -383,7 +362,7 @@ impl<T> ButtonDetails<T> {
Self {
content: ButtonContent::Icon(icon),
duration: None,
with_outline: true,
with_outline: false,
with_arms: false,
fixed_width: None,
offset: Offset::zero(),
@ -397,51 +376,42 @@ impl<T> ButtonDetails<T> {
/// Cross-style-icon cancel button with no outline.
pub fn cancel_icon() -> Self {
Self::icon(theme::ICON_CANCEL)
.with_no_outline()
.with_offset(Offset::new(2, -2))
Self::icon(theme::ICON_CANCEL).with_offset(Offset::new(3, -3))
}
/// Left arrow to signal going back. No outline.
pub fn left_arrow_icon() -> Self {
Self::icon(theme::ICON_ARROW_LEFT)
.with_no_outline()
.with_offset(Offset::new(1, -1))
Self::icon(theme::ICON_ARROW_LEFT).with_offset(Offset::new(4, -3))
}
/// Right arrow to signal going forward. No outline.
pub fn right_arrow_icon() -> Self {
Self::icon(theme::ICON_ARROW_RIGHT)
.with_no_outline()
.with_offset(Offset::new(-1, -1))
Self::icon(theme::ICON_ARROW_RIGHT).with_offset(Offset::new(-4, -3))
}
/// Up arrow to signal paginating back. No outline. Offsetted little right
/// to not be on the boundary.
pub fn up_arrow_icon() -> Self {
Self::icon(theme::ICON_ARROW_UP)
.with_no_outline()
.with_offset(Offset::new(2, -3))
Self::icon(theme::ICON_ARROW_UP).with_offset(Offset::new(2, -3))
}
/// Down arrow to signal paginating forward. Takes half the screen's width
pub fn down_arrow_icon_wide() -> Self {
Self::icon(theme::ICON_ARROW_DOWN).with_fixed_width(HALF_SCREEN_BUTTON_WIDTH)
Self::icon(theme::ICON_ARROW_DOWN)
.with_outline(true)
.with_fixed_width(HALF_SCREEN_BUTTON_WIDTH)
}
/// Up arrow to signal paginating back. Takes half the screen's width
pub fn up_arrow_icon_wide() -> Self {
Self::icon(theme::ICON_ARROW_UP).with_fixed_width(HALF_SCREEN_BUTTON_WIDTH)
Self::icon(theme::ICON_ARROW_UP)
.with_outline(true)
.with_fixed_width(HALF_SCREEN_BUTTON_WIDTH)
}
/// Icon of a bin to signal deleting.
pub fn bin_icon() -> Self {
Self::icon(theme::ICON_BIN).with_no_outline()
}
/// No outline around the button.
pub fn with_no_outline(mut self) -> Self {
self.with_outline = false;
/// Possible outline around the button.
pub fn with_outline(mut self, outline: bool) -> Self {
self.with_outline = outline;
self
}

View File

@ -106,9 +106,10 @@ pub const CHECKLIST_CURRENT_OFFSET: Offset = Offset::x(3);
// with empty LEFT column and BOTTOM row.
pub const BUTTON_CONTENT_HEIGHT: i16 = 7;
pub const BUTTON_OUTLINE: i16 = 3;
pub const BUTTON_ARMS: i16 = 2;
pub const BUTTON_HEIGHT: i16 = BUTTON_CONTENT_HEIGHT + 2 * BUTTON_OUTLINE;
pub const BUTTON_ICON_WIDTH: i16 = BUTTON_HEIGHT;
pub const TITLE_AREA_HEIGHT: i16 = 12;
pub const ARMS_MARGIN: i16 = 2;
// How many pixels should be between text and icons.
pub const ELLIPSIS_ICON_MARGIN: i16 = 4;