mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-18 11:21:11 +00:00
feat(core/rust): randomize letter choice positions in recovery word entry + randomize the order of words
[no changelog]
This commit is contained in:
parent
8c7ad72062
commit
eacc1a3826
@ -12,7 +12,7 @@ pub fn shuffle<T>(slice: &mut [T]) {
|
|||||||
|
|
||||||
/// Returns a random number in the range [min, max].
|
/// Returns a random number in the range [min, max].
|
||||||
pub fn uniform_between(min: u32, max: u32) -> u32 {
|
pub fn uniform_between(min: u32, max: u32) -> u32 {
|
||||||
assert!(max > min);
|
assert!(max >= min);
|
||||||
uniform(max - min + 1) + min
|
uniform(max - min + 1) + min
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
strutil::StringType,
|
strutil::StringType,
|
||||||
trezorhal::wordlist::Wordlist,
|
trezorhal::{random, wordlist::Wordlist},
|
||||||
ui::{
|
ui::{
|
||||||
component::{text::common::TextBox, Child, Component, ComponentExt, Event, EventCtx},
|
component::{text::common::TextBox, Child, Component, ComponentExt, Event, EventCtx},
|
||||||
geometry::Rect,
|
geometry::Rect,
|
||||||
@ -9,7 +9,7 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::super::{theme, ButtonLayout, ChangingTextLine, ChoiceFactory, ChoiceItem, ChoicePage};
|
use super::super::{theme, ButtonLayout, ChangingTextLine, ChoiceFactory, ChoiceItem, ChoicePage};
|
||||||
use heapless::String;
|
use heapless::{String, Vec};
|
||||||
|
|
||||||
enum WordlistAction {
|
enum WordlistAction {
|
||||||
Letter(char),
|
Letter(char),
|
||||||
@ -18,7 +18,6 @@ enum WordlistAction {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const MAX_WORD_LENGTH: usize = 10;
|
const MAX_WORD_LENGTH: usize = 10;
|
||||||
const MAX_LETTERS_LENGTH: usize = 26;
|
|
||||||
|
|
||||||
/// Offer words when there will be fewer of them than this
|
/// Offer words when there will be fewer of them than this
|
||||||
const OFFER_WORDS_THRESHOLD: usize = 10;
|
const OFFER_WORDS_THRESHOLD: usize = 10;
|
||||||
@ -31,6 +30,11 @@ const INITIAL_PAGE_COUNTER: usize = DELETE_INDEX + 1;
|
|||||||
|
|
||||||
const PROMPT: &str = "_";
|
const PROMPT: &str = "_";
|
||||||
|
|
||||||
|
/// Choosing random choice index, disregarding DELETE option
|
||||||
|
fn get_random_position(num_choices: usize) -> usize {
|
||||||
|
random::uniform_between(INITIAL_PAGE_COUNTER as u32, (num_choices - 1) as u32) as usize
|
||||||
|
}
|
||||||
|
|
||||||
/// Type of the wordlist, deciding the list of words to be used
|
/// Type of the wordlist, deciding the list of words to be used
|
||||||
#[derive(Clone, Copy)]
|
#[derive(Clone, Copy)]
|
||||||
pub enum WordlistType {
|
pub enum WordlistType {
|
||||||
@ -41,6 +45,8 @@ pub enum WordlistType {
|
|||||||
struct ChoiceFactoryWordlist {
|
struct ChoiceFactoryWordlist {
|
||||||
wordlist: Wordlist,
|
wordlist: Wordlist,
|
||||||
offer_words: bool,
|
offer_words: bool,
|
||||||
|
/// We want to randomize the order in which we show the words
|
||||||
|
word_random_order: Vec<usize, OFFER_WORDS_THRESHOLD>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChoiceFactoryWordlist {
|
impl ChoiceFactoryWordlist {
|
||||||
@ -51,9 +57,21 @@ impl ChoiceFactoryWordlist {
|
|||||||
}
|
}
|
||||||
.filter_prefix(prefix);
|
.filter_prefix(prefix);
|
||||||
let offer_words = wordlist.len() < OFFER_WORDS_THRESHOLD;
|
let offer_words = wordlist.len() < OFFER_WORDS_THRESHOLD;
|
||||||
|
let word_random_order: Vec<usize, OFFER_WORDS_THRESHOLD> = if offer_words {
|
||||||
|
// Filling slice with numbers 0..wordlist.len() and shuffling them
|
||||||
|
let slice = &mut [0; OFFER_WORDS_THRESHOLD][..wordlist.len()];
|
||||||
|
for (i, item) in slice.iter_mut().enumerate() {
|
||||||
|
*item = i;
|
||||||
|
}
|
||||||
|
random::shuffle(slice);
|
||||||
|
slice.iter().copied().collect()
|
||||||
|
} else {
|
||||||
|
Vec::new()
|
||||||
|
};
|
||||||
Self {
|
Self {
|
||||||
wordlist,
|
wordlist,
|
||||||
offer_words,
|
offer_words,
|
||||||
|
word_random_order,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -82,7 +100,9 @@ impl<T: StringType + Clone> ChoiceFactory<T> for ChoiceFactoryWordlist {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
if self.offer_words {
|
if self.offer_words {
|
||||||
let word = self.wordlist.get(choice_index - 1).unwrap_or_default();
|
// Taking a random (but always the same) word on this position
|
||||||
|
let index = self.word_random_order[choice_index - 1];
|
||||||
|
let word = self.wordlist.get(index).unwrap_or_default();
|
||||||
(
|
(
|
||||||
ChoiceItem::new(word, ButtonLayout::default_three_icons()),
|
ChoiceItem::new(word, ButtonLayout::default_three_icons()),
|
||||||
WordlistAction::Word(word),
|
WordlistAction::Word(word),
|
||||||
@ -116,12 +136,13 @@ where
|
|||||||
{
|
{
|
||||||
pub fn new(wordlist_type: WordlistType) -> Self {
|
pub fn new(wordlist_type: WordlistType) -> Self {
|
||||||
let choices = ChoiceFactoryWordlist::new(wordlist_type, "");
|
let choices = ChoiceFactoryWordlist::new(wordlist_type, "");
|
||||||
|
let choices_count = <ChoiceFactoryWordlist as ChoiceFactory<T>>::count(&choices);
|
||||||
Self {
|
Self {
|
||||||
// Starting at second page because of DELETE option
|
// Starting at random letter position
|
||||||
choice_page: ChoicePage::new(choices)
|
choice_page: ChoicePage::new(choices)
|
||||||
.with_incomplete(true)
|
.with_incomplete(true)
|
||||||
.with_carousel(true)
|
.with_carousel(true)
|
||||||
.with_initial_page_counter(INITIAL_PAGE_COUNTER),
|
.with_initial_page_counter(get_random_position(choices_count)),
|
||||||
chosen_letters: Child::new(ChangingTextLine::center_mono(String::from(PROMPT))),
|
chosen_letters: Child::new(ChangingTextLine::center_mono(String::from(PROMPT))),
|
||||||
textbox: TextBox::empty(),
|
textbox: TextBox::empty(),
|
||||||
offer_words: false,
|
offer_words: false,
|
||||||
@ -140,14 +161,18 @@ where
|
|||||||
self.update_chosen_letters(ctx);
|
self.update_chosen_letters(ctx);
|
||||||
let new_choices = self.get_current_choices();
|
let new_choices = self.get_current_choices();
|
||||||
self.offer_words = new_choices.offer_words;
|
self.offer_words = new_choices.offer_words;
|
||||||
|
// Starting at the random position in case of letters and at the beginning in
|
||||||
|
// case of words
|
||||||
|
let new_page_counter = if self.offer_words {
|
||||||
|
INITIAL_PAGE_COUNTER
|
||||||
|
} else {
|
||||||
|
let choices_count = <ChoiceFactoryWordlist as ChoiceFactory<T>>::count(&new_choices);
|
||||||
|
get_random_position(choices_count)
|
||||||
|
};
|
||||||
// Not using carousel in case of words, as that looks weird in case
|
// Not using carousel in case of words, as that looks weird in case
|
||||||
// there is only one word to choose from.
|
// there is only one word to choose from.
|
||||||
self.choice_page.reset(
|
self.choice_page
|
||||||
ctx,
|
.reset(ctx, new_choices, Some(new_page_counter), !self.offer_words);
|
||||||
new_choices,
|
|
||||||
Some(INITIAL_PAGE_COUNTER),
|
|
||||||
!self.offer_words,
|
|
||||||
);
|
|
||||||
ctx.request_paint();
|
ctx.request_paint();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user