mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
00341cd3b3
* Initial implementation of new SrpInput component This new version of the SrpInput component uses a separate field for each word of the SRP. Only one field can be revealed at a time, making it less likely that it gets accidentally revealed to somebody. * Fix copy mistakes * Move container div from 'create vault' to 'srp-input', and setup grid layout * Increase size of title * Remove hard-coded width in Storybook to allow testing different viewport sizes * Improve layout * Improve margins * Update dropdown text * Expand SRP input section * Remove unused localized messages * Update dropdown option names in unit tests * Replace checkbox with show/hide toggle * Remove unused localized message * Fix 'data-testid' prop name * Fix e2e test imports using paste * Use 'ActionableMessage' component for error message * Convert error popover to actionable message * Add tip about pasting the SRP * Remove invalid prop The "info" style of `ActionableMessage` is the default, so no type is required. * Use more readable test convenience methods The method `toBeInTheDocument()` is now used over `not.toBeNull()` to improve the readability of tests. Likewise, the convenience method `.clear` is now used to clear fields rather than manually entering the key combination to clear a field. * Fix misspelled word
236 lines
7.5 KiB
JavaScript
236 lines
7.5 KiB
JavaScript
import { ethers } from 'ethers';
|
|
import React, { useCallback, useState } from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import { useI18nContext } from '../../../hooks/useI18nContext';
|
|
import TextField from '../../ui/text-field';
|
|
import { clearClipboard } from '../../../helpers/utils/util';
|
|
import ActionableMessage from '../../ui/actionable-message';
|
|
import Dropdown from '../../ui/dropdown';
|
|
import Typography from '../../ui/typography';
|
|
import ShowHideToggle from '../../ui/show-hide-toggle';
|
|
import { TYPOGRAPHY } from '../../../helpers/constants/design-system';
|
|
import { parseSecretRecoveryPhrase } from './parse-secret-recovery-phrase';
|
|
|
|
const { isValidMnemonic } = ethers.utils;
|
|
|
|
const defaultNumberOfWords = 12;
|
|
|
|
export default function SrpInput({ onChange }) {
|
|
const [srpError, setSrpError] = useState('');
|
|
const [pasteFailed, setPasteFailed] = useState(false);
|
|
const [draftSrp, setDraftSrp] = useState(
|
|
new Array(defaultNumberOfWords).fill(''),
|
|
);
|
|
const [showSrp, setShowSrp] = useState(
|
|
new Array(defaultNumberOfWords).fill(false),
|
|
);
|
|
const [numberOfWords, setNumberOfWords] = useState(defaultNumberOfWords);
|
|
|
|
const t = useI18nContext();
|
|
|
|
const onSrpChange = useCallback(
|
|
(newDraftSrp) => {
|
|
let newSrpError = '';
|
|
const joinedDraftSrp = newDraftSrp.join(' ');
|
|
|
|
if (newDraftSrp.some((word) => word !== '')) {
|
|
if (newDraftSrp.some((word) => word === '')) {
|
|
newSrpError = t('seedPhraseReq');
|
|
} else if (!isValidMnemonic(joinedDraftSrp)) {
|
|
newSrpError = t('invalidSeedPhrase');
|
|
}
|
|
}
|
|
|
|
setDraftSrp(newDraftSrp);
|
|
setSrpError(newSrpError);
|
|
onChange(newSrpError ? '' : joinedDraftSrp);
|
|
},
|
|
[setDraftSrp, setSrpError, t, onChange],
|
|
);
|
|
|
|
const toggleShowSrp = useCallback((index) => {
|
|
setShowSrp((currentShowSrp) => {
|
|
const newShowSrp = currentShowSrp.slice();
|
|
if (newShowSrp[index]) {
|
|
newShowSrp[index] = false;
|
|
} else {
|
|
newShowSrp.fill(false);
|
|
newShowSrp[index] = true;
|
|
}
|
|
return newShowSrp;
|
|
});
|
|
}, []);
|
|
|
|
const onSrpWordChange = useCallback(
|
|
(index, newWord) => {
|
|
if (pasteFailed) {
|
|
setPasteFailed(false);
|
|
}
|
|
const newSrp = draftSrp.slice();
|
|
newSrp[index] = newWord.trim();
|
|
onSrpChange(newSrp);
|
|
},
|
|
[draftSrp, onSrpChange, pasteFailed],
|
|
);
|
|
|
|
const onSrpPaste = useCallback(
|
|
(rawSrp) => {
|
|
const parsedSrp = parseSecretRecoveryPhrase(rawSrp);
|
|
let newDraftSrp = parsedSrp.split(' ');
|
|
|
|
if (newDraftSrp.length > 24) {
|
|
setPasteFailed(true);
|
|
return;
|
|
} else if (pasteFailed) {
|
|
setPasteFailed(false);
|
|
}
|
|
|
|
let newNumberOfWords = numberOfWords;
|
|
if (newDraftSrp.length !== numberOfWords) {
|
|
if (newDraftSrp.length < 12) {
|
|
newNumberOfWords = 12;
|
|
} else if (newDraftSrp.length % 3 === 0) {
|
|
newNumberOfWords = newDraftSrp.length;
|
|
} else {
|
|
newNumberOfWords =
|
|
newDraftSrp.length + (3 - (newDraftSrp.length % 3));
|
|
}
|
|
setNumberOfWords(newNumberOfWords);
|
|
}
|
|
|
|
if (newDraftSrp.length < newNumberOfWords) {
|
|
newDraftSrp = newDraftSrp.concat(
|
|
new Array(newNumberOfWords - newDraftSrp.length).fill(''),
|
|
);
|
|
}
|
|
setShowSrp(new Array(newNumberOfWords).fill(false));
|
|
onSrpChange(newDraftSrp);
|
|
clearClipboard();
|
|
},
|
|
[numberOfWords, onSrpChange, pasteFailed, setPasteFailed],
|
|
);
|
|
|
|
const numberOfWordsOptions = [];
|
|
for (let i = 12; i <= 24; i += 3) {
|
|
numberOfWordsOptions.push({
|
|
name: t('srpInputNumberOfWords', [`${i}`]),
|
|
value: `${i}`,
|
|
});
|
|
}
|
|
|
|
return (
|
|
<div className="import-srp__container">
|
|
<label className="import-srp__srp-label">
|
|
<Typography variant={TYPOGRAPHY.H4}>
|
|
{t('secretRecoveryPhrase')}
|
|
</Typography>
|
|
</label>
|
|
<ActionableMessage
|
|
className="import-srp__paste-tip"
|
|
iconFillColor="#037dd6" // This is `--color-info-default`
|
|
message={t('srpPasteTip')}
|
|
useIcon
|
|
/>
|
|
<Dropdown
|
|
className="import-srp__number-of-words-dropdown"
|
|
onChange={(newSelectedOption) => {
|
|
const newNumberOfWords = parseInt(newSelectedOption, 10);
|
|
if (Number.isNaN(newNumberOfWords)) {
|
|
throw new Error('Unable to parse option as integer');
|
|
}
|
|
|
|
let newDraftSrp = draftSrp.slice(0, newNumberOfWords);
|
|
if (newDraftSrp.length < newNumberOfWords) {
|
|
newDraftSrp = newDraftSrp.concat(
|
|
new Array(newNumberOfWords - newDraftSrp.length).fill(''),
|
|
);
|
|
}
|
|
setNumberOfWords(newNumberOfWords);
|
|
setShowSrp(new Array(newNumberOfWords).fill(false));
|
|
onSrpChange(newDraftSrp);
|
|
}}
|
|
options={numberOfWordsOptions}
|
|
selectedOption={`${numberOfWords}`}
|
|
/>
|
|
<div className="import-srp__srp">
|
|
{[...Array(numberOfWords).keys()].map((index) => {
|
|
const id = `import-srp__srp-word-${index}`;
|
|
return (
|
|
<div key={index} className="import-srp__srp-word">
|
|
<label htmlFor={id} className="import-srp__srp-word-label">
|
|
<Typography>{`${index + 1}.`}</Typography>
|
|
</label>
|
|
<TextField
|
|
id={id}
|
|
data-testid={id}
|
|
type={showSrp[index] ? 'text' : 'password'}
|
|
onChange={(e) => {
|
|
e.preventDefault();
|
|
onSrpWordChange(index, e.target.value);
|
|
}}
|
|
value={draftSrp[index]}
|
|
autoComplete="off"
|
|
onPaste={(event) => {
|
|
const newSrp = event.clipboardData.getData('text');
|
|
|
|
if (newSrp.trim().match(/\s/u)) {
|
|
event.preventDefault();
|
|
onSrpPaste(newSrp);
|
|
} else {
|
|
onSrpWordChange(index, newSrp);
|
|
}
|
|
}}
|
|
/>
|
|
<ShowHideToggle
|
|
id={`${id}-checkbox`}
|
|
ariaLabelHidden={t('srpWordHidden')}
|
|
ariaLabelShown={t('srpWordShown')}
|
|
shown={showSrp[index]}
|
|
data-testid={`${id}-checkbox`}
|
|
onChange={() => toggleShowSrp(index)}
|
|
title={t('srpToggleShow')}
|
|
/>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
{srpError ? (
|
|
<ActionableMessage
|
|
className="import-srp__srp-error"
|
|
iconFillColor="#d73a49" // This is `--color-error-default`
|
|
message={srpError}
|
|
type="danger"
|
|
useIcon
|
|
/>
|
|
) : null}
|
|
{pasteFailed ? (
|
|
<ActionableMessage
|
|
className="import-srp__srp-too-many-words-error"
|
|
iconFillColor="#d73a49" // This is `--color-error-default`
|
|
message={t('srpPasteFailedTooManyWords')}
|
|
primaryAction={{
|
|
label: t('dismiss'),
|
|
onClick: () => setPasteFailed(false),
|
|
}}
|
|
type="danger"
|
|
useIcon
|
|
/>
|
|
) : null}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
SrpInput.propTypes = {
|
|
/**
|
|
* Event handler for SRP changes.
|
|
*
|
|
* This is only called with a valid, well-formated (i.e. exactly one space
|
|
* between each word) SRP or with an empty string.
|
|
*
|
|
* This is called each time the draft SRP is updated. If the draft SRP is
|
|
* valid, this is called with a well-formatted version of that draft SRP.
|
|
* Otherwise, this is called with an empty string.
|
|
*/
|
|
onChange: PropTypes.func.isRequired,
|
|
};
|