mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-17 19:00:58 +00:00
refactor(core/ui): replace confirm_hex, clarify API
Truncation options were removed. Subtitle distinct from description was removed. confirm_hex was replaced by confirm_blob. You should use confirm_blob when displaying data that is not human readable and can be broken at any character. Also it is now possible to pass bytes, which are automatically converted to hex. For displaying addresses, a separate confirm_address is introduced, which simply delegates to confirm_blob, but has a more limited signature. Analogously, there is confirm_text for text data (should maybe be used in many places where we currently use confirm_metadata) and a specialized confirm_amount.
This commit is contained in:
parent
6d15e90ed3
commit
dcc38f5267
@ -43,6 +43,9 @@ if False:
|
|||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
"confirm_action",
|
"confirm_action",
|
||||||
|
"confirm_address",
|
||||||
|
"confirm_text",
|
||||||
|
"confirm_amount",
|
||||||
"confirm_reset_device",
|
"confirm_reset_device",
|
||||||
"confirm_backup",
|
"confirm_backup",
|
||||||
"confirm_path_warning",
|
"confirm_path_warning",
|
||||||
@ -56,7 +59,7 @@ __all__ = (
|
|||||||
"show_warning",
|
"show_warning",
|
||||||
"confirm_output",
|
"confirm_output",
|
||||||
"confirm_decred_sstx_submission",
|
"confirm_decred_sstx_submission",
|
||||||
"confirm_hex",
|
"confirm_blob",
|
||||||
"confirm_properties",
|
"confirm_properties",
|
||||||
"confirm_total",
|
"confirm_total",
|
||||||
"confirm_total_ethereum",
|
"confirm_total_ethereum",
|
||||||
@ -356,14 +359,13 @@ async def show_address(
|
|||||||
def show_pubkey(
|
def show_pubkey(
|
||||||
ctx: wire.Context, pubkey: str, title: str = "Confirm public key"
|
ctx: wire.Context, pubkey: str, title: str = "Confirm public key"
|
||||||
) -> Awaitable[None]:
|
) -> Awaitable[None]:
|
||||||
return confirm_hex(
|
return confirm_blob(
|
||||||
ctx,
|
ctx,
|
||||||
br_type="show_pubkey",
|
br_type="show_pubkey",
|
||||||
title="Confirm public key",
|
title="Confirm public key",
|
||||||
data=pubkey,
|
data=pubkey,
|
||||||
br_code=ButtonRequestType.PublicKey,
|
br_code=ButtonRequestType.PublicKey,
|
||||||
icon=ui.ICON_RECEIVE,
|
icon=ui.ICON_RECEIVE,
|
||||||
truncate=True, # should fit?
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -521,64 +523,161 @@ async def confirm_decred_sstx_submission(
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def confirm_hex(
|
async def confirm_blob(
|
||||||
ctx: wire.GenericContext,
|
ctx: wire.GenericContext,
|
||||||
br_type: str,
|
br_type: str,
|
||||||
title: str,
|
title: str,
|
||||||
data: str,
|
data: bytes | str,
|
||||||
subtitle: str | None = None,
|
|
||||||
description: str | None = None,
|
description: str | None = None,
|
||||||
br_code: ButtonRequestType = ButtonRequestType.Other,
|
br_code: ButtonRequestType = ButtonRequestType.Other,
|
||||||
icon: str = ui.ICON_SEND, # TODO cleanup @ redesign
|
icon: str = ui.ICON_SEND, # TODO cleanup @ redesign
|
||||||
icon_color: int = ui.GREEN, # TODO cleanup @ redesign
|
icon_color: int = ui.GREEN, # TODO cleanup @ redesign
|
||||||
color_description: int = ui.FG, # TODO cleanup @ redesign
|
|
||||||
width: int | None = MONO_HEX_PER_LINE,
|
|
||||||
width_paginated: int = MONO_HEX_PER_LINE - 2,
|
|
||||||
truncate: bool = False,
|
|
||||||
truncate_middle: bool = False,
|
|
||||||
truncate_ellipsis: str = "...",
|
|
||||||
) -> None:
|
) -> None:
|
||||||
if truncate:
|
"""Confirm data blob.
|
||||||
|
|
||||||
|
Applicable for public keys, signatures, hashes. In general, any kind of
|
||||||
|
data that is not human-readable, and can be wrapped at any character.
|
||||||
|
|
||||||
|
For addresses, use `confirm_address`.
|
||||||
|
|
||||||
|
Displays in monospace font. Paginates automatically.
|
||||||
|
If data is provided as bytes or bytearray, it is converted to hex.
|
||||||
|
"""
|
||||||
|
if isinstance(data, (bytes, bytearray)):
|
||||||
|
data_str = hexlify(data).decode()
|
||||||
|
else:
|
||||||
|
data_str = data
|
||||||
|
|
||||||
|
span = Span()
|
||||||
|
lines = 0
|
||||||
|
if description is not None:
|
||||||
|
span.reset(description, 0, ui.NORMAL)
|
||||||
|
lines += span.count_lines()
|
||||||
|
data_lines = (len(data_str) + MONO_HEX_PER_LINE - 1) // MONO_HEX_PER_LINE
|
||||||
|
lines += data_lines
|
||||||
|
|
||||||
|
if lines <= TEXT_MAX_LINES:
|
||||||
text = Text(title, icon, icon_color, new_lines=False)
|
text = Text(title, icon, icon_color, new_lines=False)
|
||||||
description_lines = 0
|
|
||||||
if subtitle is not None:
|
|
||||||
description_lines += Span(subtitle, 0, ui.BOLD).count_lines()
|
|
||||||
text.bold(subtitle)
|
|
||||||
text.br()
|
|
||||||
if description is not None:
|
if description is not None:
|
||||||
description_lines += Span(description, 0, ui.NORMAL).count_lines()
|
text.normal(description)
|
||||||
text.content.extend((ui.NORMAL, color_description, description, ui.FG))
|
|
||||||
text.br()
|
text.br()
|
||||||
if width is not None:
|
|
||||||
text.mono(
|
# special case:
|
||||||
*_truncate_hex(
|
if len(data_str) % 16 == 0:
|
||||||
data,
|
# sanity checks:
|
||||||
lines=TEXT_MAX_LINES - description_lines,
|
# (a) we must not exceed MONO_HEX_PER_LINE
|
||||||
width=width,
|
assert MONO_HEX_PER_LINE > 16
|
||||||
middle=truncate_middle,
|
# (b) we must not increase number of lines
|
||||||
ellipsis=truncate_ellipsis,
|
assert (len(data_str) // 16) <= data_lines
|
||||||
)
|
# the above holds true for MONO_HEX_PER_LINE == 18 and TEXT_MAX_LINES == 5
|
||||||
)
|
per_line = 16
|
||||||
|
|
||||||
else:
|
else:
|
||||||
text.mono(data)
|
per_line = MONO_HEX_PER_LINE
|
||||||
|
text.mono(ui.FG, *chunks_intersperse(data_str, per_line))
|
||||||
content: ui.Layout = Confirm(text)
|
content: ui.Layout = Confirm(text)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
para = []
|
para = []
|
||||||
if subtitle is not None:
|
|
||||||
para.append((ui.BOLD, subtitle))
|
|
||||||
if description is not None:
|
if description is not None:
|
||||||
assert (
|
|
||||||
color_description == ui.FG
|
|
||||||
) # only ethereum uses this and it truncates
|
|
||||||
para.append((ui.NORMAL, description))
|
para.append((ui.NORMAL, description))
|
||||||
if width is not None:
|
para.extend((ui.MONO, line) for line in chunks(data_str, MONO_HEX_PER_LINE - 2))
|
||||||
para.extend((ui.MONO, line) for line in chunks(data, width_paginated))
|
|
||||||
else:
|
|
||||||
para.append((ui.MONO, data))
|
|
||||||
content = paginate_paragraphs(para, title, icon, icon_color)
|
content = paginate_paragraphs(para, title, icon, icon_color)
|
||||||
await raise_if_cancelled(interact(ctx, content, br_type, br_code))
|
await raise_if_cancelled(interact(ctx, content, br_type, br_code))
|
||||||
|
|
||||||
|
|
||||||
|
def confirm_address(
|
||||||
|
ctx: wire.GenericContext,
|
||||||
|
title: str,
|
||||||
|
address: str,
|
||||||
|
description: str | None = "Address:",
|
||||||
|
br_type: str = "confirm_address",
|
||||||
|
br_code: ButtonRequestType = ButtonRequestType.Other,
|
||||||
|
icon: str = ui.ICON_SEND, # TODO cleanup @ redesign
|
||||||
|
icon_color: int = ui.GREEN, # TODO cleanup @ redesign
|
||||||
|
) -> Awaitable[None]:
|
||||||
|
# TODO clarify API - this should be pretty limited to support mainly confirming
|
||||||
|
# destinations and similar
|
||||||
|
return confirm_blob(
|
||||||
|
ctx,
|
||||||
|
br_type=br_type,
|
||||||
|
title=title,
|
||||||
|
data=address,
|
||||||
|
description=description,
|
||||||
|
br_code=br_code,
|
||||||
|
icon=icon,
|
||||||
|
icon_color=icon_color,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def confirm_text(
|
||||||
|
ctx: wire.GenericContext,
|
||||||
|
br_type: str,
|
||||||
|
title: str,
|
||||||
|
data: str,
|
||||||
|
description: str | None = None,
|
||||||
|
br_code: ButtonRequestType = ButtonRequestType.Other,
|
||||||
|
icon: str = ui.ICON_SEND, # TODO cleanup @ redesign
|
||||||
|
icon_color: int = ui.GREEN, # TODO cleanup @ redesign
|
||||||
|
) -> None:
|
||||||
|
"""Confirm textual data.
|
||||||
|
|
||||||
|
Applicable for human-readable strings, numbers, date/time values etc.
|
||||||
|
|
||||||
|
For amounts, use `confirm_amount`.
|
||||||
|
|
||||||
|
Displays in bold font. Paginates automatically.
|
||||||
|
"""
|
||||||
|
span = Span()
|
||||||
|
lines = 0
|
||||||
|
if description is not None:
|
||||||
|
span.reset(description, 0, ui.NORMAL)
|
||||||
|
lines += span.count_lines()
|
||||||
|
span.reset(data, 0, ui.BOLD)
|
||||||
|
lines += span.count_lines()
|
||||||
|
|
||||||
|
if lines <= TEXT_MAX_LINES:
|
||||||
|
text = Text(title, icon, icon_color, new_lines=False)
|
||||||
|
if description is not None:
|
||||||
|
text.normal(description)
|
||||||
|
text.br()
|
||||||
|
text.bold(data)
|
||||||
|
content: ui.Layout = Confirm(text)
|
||||||
|
|
||||||
|
else:
|
||||||
|
para = []
|
||||||
|
if description is not None:
|
||||||
|
para.append((ui.NORMAL, description))
|
||||||
|
para.append((ui.BOLD, data))
|
||||||
|
content = paginate_paragraphs(para, title, icon, icon_color)
|
||||||
|
await raise_if_cancelled(interact(ctx, content, br_type, br_code))
|
||||||
|
|
||||||
|
|
||||||
|
def confirm_amount(
|
||||||
|
ctx: wire.GenericContext,
|
||||||
|
title: str,
|
||||||
|
amount: str,
|
||||||
|
description: str = "Amount:",
|
||||||
|
br_type: str = "confirm_amount",
|
||||||
|
br_code: ButtonRequestType = ButtonRequestType.Other,
|
||||||
|
icon: str = ui.ICON_SEND, # TODO cleanup @ redesign
|
||||||
|
icon_color: int = ui.GREEN, # TODO cleanup @ redesign
|
||||||
|
) -> Awaitable[None]:
|
||||||
|
"""Confirm amount."""
|
||||||
|
# TODO clarify API - this should be pretty limited to support mainly confirming
|
||||||
|
# destinations and similar
|
||||||
|
return confirm_text(
|
||||||
|
ctx,
|
||||||
|
br_type=br_type,
|
||||||
|
title=title,
|
||||||
|
data=amount,
|
||||||
|
description=description,
|
||||||
|
br_code=br_code,
|
||||||
|
icon=icon,
|
||||||
|
icon_color=icon_color,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
_SCREEN_FULL_THRESHOLD = const(2)
|
_SCREEN_FULL_THRESHOLD = const(2)
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user