mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
9acd4b4ea1
* feat(srp): add a quiz to the SRP reveal * fixed the popover header centering * lint fixes * converted from `ui/components/ui/popover` to `ui/components/component-library/modal` * responded to @darkwing review * added unit tests * renamed the folder to 'srp-quiz-modal' * responded to Monte's review * using i18n-helper in the test suite * small improvement to JSXDict comments * wrote a new webdriver.holdMouseDownOnElement() to assist with testing the "Hold to reveal SRP" button * Updating layout and some storybook naming and migrating to tsx * Apply suggestions from @georgewrmarshall Co-authored-by: George Marshall <george.marshall@consensys.net> * Unit test searches by data-testid instead of by text * new layout and copy for the Settings->Security page * now with 100% test coverage for /ui/pages/settings/security-tab fixes #16871 fixes #18140 * e2e tests to reveal SRP after quiz * e2e- Fix lint, remove unneeded extras * @coreyjanssen and @georgewrmarshall compromise Co-authored-by: George Marshall <george.marshall@consensys.net> Co-authored-by: Corey Janssen <corey.janssen@consensys.net> * trying isRequired again * transparent background on PNG * [e2e] moving functions to helpers and adding testid for SRP reveal quiz (#19481) * moving functions to helpers and adding testid * fix lint error * took out the IPFS gateway fixes * lint fix * translations of SRP Reveal Quiz * new Spanish translation from Guto * Update describe for e2e tests * Apply suggestion from @georgewrmarshall Co-authored-by: George Marshall <george.marshall@consensys.net> * fixed the Tab key problem --------- Co-authored-by: georgewrmarshall <george.marshall@consensys.net> Co-authored-by: Plasma Corral <32695229+plasmacorral@users.noreply.github.com> Co-authored-by: Corey Janssen <corey.janssen@consensys.net>
385 lines
11 KiB
JavaScript
385 lines
11 KiB
JavaScript
import React, { useContext, useEffect, useState } from 'react';
|
|
import { useSelector, useDispatch } from 'react-redux';
|
|
import { useHistory } from 'react-router-dom';
|
|
import qrCode from 'qrcode-generator';
|
|
import { requestRevealSeedWords, showModal } from '../../store/actions';
|
|
import ExportTextContainer from '../../components/ui/export-text-container';
|
|
import { getMostRecentOverviewPage } from '../../ducks/history/history';
|
|
import {
|
|
MetaMetricsEventCategory,
|
|
MetaMetricsEventKeyType,
|
|
MetaMetricsEventName,
|
|
} from '../../../shared/constants/metametrics';
|
|
import {
|
|
TextVariant,
|
|
SEVERITIES,
|
|
Size,
|
|
BLOCK_SIZES,
|
|
JustifyContent,
|
|
AlignItems,
|
|
DISPLAY,
|
|
} from '../../helpers/constants/design-system';
|
|
|
|
import Box from '../../components/ui/box';
|
|
import {
|
|
Text,
|
|
Label,
|
|
BannerAlert,
|
|
Button,
|
|
TextField,
|
|
HelpText,
|
|
BUTTON_VARIANT,
|
|
TEXT_FIELD_SIZES,
|
|
TEXT_FIELD_TYPES,
|
|
BUTTON_SIZES,
|
|
} from '../../components/component-library';
|
|
import { useI18nContext } from '../../hooks/useI18nContext';
|
|
import { MetaMetricsContext } from '../../contexts/metametrics';
|
|
import ZENDESK_URLS from '../../helpers/constants/zendesk-url';
|
|
import { Tabs, Tab } from '../../components/ui/tabs';
|
|
|
|
const PASSWORD_PROMPT_SCREEN = 'PASSWORD_PROMPT_SCREEN';
|
|
const REVEAL_SEED_SCREEN = 'REVEAL_SEED_SCREEN';
|
|
|
|
const RevealSeedPage = () => {
|
|
const history = useHistory();
|
|
const dispatch = useDispatch();
|
|
const t = useI18nContext();
|
|
const trackEvent = useContext(MetaMetricsContext);
|
|
|
|
const [screen, setScreen] = useState(PASSWORD_PROMPT_SCREEN);
|
|
const [password, setPassword] = useState('');
|
|
const [seedWords, setSeedWords] = useState(null);
|
|
const [completedLongPress, setCompletedLongPress] = useState(false);
|
|
const [error, setError] = useState(null);
|
|
const mostRecentOverviewPage = useSelector(getMostRecentOverviewPage);
|
|
|
|
useEffect(() => {
|
|
const passwordBox = document.getElementById('password-box');
|
|
if (passwordBox) {
|
|
passwordBox.focus();
|
|
}
|
|
}, []);
|
|
|
|
const renderQR = () => {
|
|
const qrImage = qrCode(0, 'L');
|
|
qrImage.addData(seedWords);
|
|
qrImage.make();
|
|
return qrImage;
|
|
};
|
|
|
|
const handleSubmit = (event) => {
|
|
event.preventDefault();
|
|
setSeedWords(null);
|
|
setCompletedLongPress(false);
|
|
setError(null);
|
|
dispatch(requestRevealSeedWords(password))
|
|
.then((revealedSeedWords) => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.KeyExportRevealed,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
setSeedWords(revealedSeedWords);
|
|
|
|
dispatch(
|
|
showModal({
|
|
name: 'HOLD_TO_REVEAL_SRP',
|
|
onLongPressed: () => {
|
|
setCompletedLongPress(true);
|
|
setScreen(REVEAL_SEED_SCREEN);
|
|
},
|
|
holdToRevealType: 'SRP',
|
|
}),
|
|
);
|
|
})
|
|
.catch((e) => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.KeyExportFailed,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
reason: e.message, // 'incorrect_password',
|
|
},
|
|
});
|
|
setError(e.message);
|
|
});
|
|
};
|
|
|
|
const renderWarning = () => {
|
|
return (
|
|
<BannerAlert severity={SEVERITIES.DANGER}>
|
|
<Text variant={TextVariant.bodyMd}>
|
|
{t('revealSeedWordsWarning', [
|
|
<Text
|
|
key="reveal-seed-words-warning-2"
|
|
variant={TextVariant.bodyMdBold}
|
|
as="strong"
|
|
>
|
|
{t('revealSeedWordsWarning2')}
|
|
</Text>,
|
|
])}
|
|
</Text>
|
|
</BannerAlert>
|
|
);
|
|
};
|
|
|
|
const renderPasswordPromptContent = () => {
|
|
return (
|
|
<form onSubmit={(event) => handleSubmit(event)}>
|
|
<Label htmlFor="password-box">{t('enterPasswordContinue')}</Label>
|
|
<TextField
|
|
inputProps={{
|
|
'data-testid': 'input-password',
|
|
}}
|
|
type={TEXT_FIELD_TYPES.PASSWORD}
|
|
placeholder={t('makeSureNoOneWatching')}
|
|
id="password-box"
|
|
size={TEXT_FIELD_SIZES.LG}
|
|
value={password}
|
|
onChange={(event) => setPassword(event.target.value)}
|
|
error={error}
|
|
width={BLOCK_SIZES.FULL}
|
|
/>
|
|
{error && <HelpText severity={SEVERITIES.DANGER}>{error}</HelpText>}
|
|
</form>
|
|
);
|
|
};
|
|
|
|
const renderRevealSeedContent = () => {
|
|
// default for SRP_VIEW_SRP_TEXT event because this is the first thing shown after rendering
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpViewSrpText,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
|
|
return (
|
|
<div>
|
|
<Tabs
|
|
defaultActiveTabName={t('revealSeedWordsText')}
|
|
onTabClick={(tabName) => {
|
|
if (tabName === 'text-seed') {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpViewSrpText,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
} else if (tabName === 'qr-srp') {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpViewsSrpQR,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
}
|
|
}}
|
|
>
|
|
<Tab
|
|
name={t('revealSeedWordsText')}
|
|
className="reveal-seed__tab"
|
|
activeClassName="reveal-seed__active-tab"
|
|
tabKey="text-seed"
|
|
>
|
|
<Label marginTop={4}>{t('yourPrivateSeedPhrase')}</Label>
|
|
<ExportTextContainer
|
|
text={seedWords}
|
|
onClickCopy={() => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.KeyExportCopied,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
copy_method: 'clipboard',
|
|
},
|
|
});
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpCopiedToClipboard,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
copy_method: 'clipboard',
|
|
},
|
|
});
|
|
}}
|
|
/>
|
|
</Tab>
|
|
<Tab
|
|
name={t('revealSeedWordsQR')}
|
|
className="reveal-seed__tab"
|
|
activeClassName="reveal-seed__active-tab"
|
|
tabKey="qr-srp"
|
|
>
|
|
<Box
|
|
display={DISPLAY.FLEX}
|
|
justifyContent={JustifyContent.center}
|
|
alignItems={AlignItems.center}
|
|
paddingTop={4}
|
|
data-testid="qr-srp"
|
|
>
|
|
<div
|
|
dangerouslySetInnerHTML={{
|
|
__html: renderQR().createTableTag(5, 15),
|
|
}}
|
|
/>
|
|
</Box>
|
|
</Tab>
|
|
</Tabs>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const renderPasswordPromptFooter = () => {
|
|
return (
|
|
<Box display={DISPLAY.FLEX} marginTop="auto" gap={4}>
|
|
<Button
|
|
width={BLOCK_SIZES.FULL}
|
|
size={Size.LG}
|
|
variant={BUTTON_VARIANT.SECONDARY}
|
|
onClick={() => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.KeyExportCanceled,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpRevealCancelled,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
history.push(mostRecentOverviewPage);
|
|
}}
|
|
>
|
|
{t('cancel')}
|
|
</Button>
|
|
<Button
|
|
width={BLOCK_SIZES.FULL}
|
|
size={Size.LG}
|
|
onClick={(event) => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.KeyExportRequested,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpRevealNextClicked,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
handleSubmit(event);
|
|
}}
|
|
disabled={password === ''}
|
|
>
|
|
{t('next')}
|
|
</Button>
|
|
</Box>
|
|
);
|
|
};
|
|
|
|
const renderRevealSeedFooter = () => {
|
|
return (
|
|
<Box marginTop="auto">
|
|
<Button
|
|
variant={BUTTON_VARIANT.SECONDARY}
|
|
width={BLOCK_SIZES.FULL}
|
|
size={Size.LG}
|
|
onClick={() => {
|
|
trackEvent({
|
|
category: MetaMetricsEventCategory.Keys,
|
|
event: MetaMetricsEventName.SrpRevealCloseClicked,
|
|
properties: {
|
|
key_type: MetaMetricsEventKeyType.Srp,
|
|
},
|
|
});
|
|
history.push(mostRecentOverviewPage);
|
|
}}
|
|
>
|
|
{t('close')}
|
|
</Button>
|
|
</Box>
|
|
);
|
|
};
|
|
|
|
const renderContent = () => {
|
|
return screen === PASSWORD_PROMPT_SCREEN || !completedLongPress
|
|
? renderPasswordPromptContent()
|
|
: renderRevealSeedContent();
|
|
};
|
|
|
|
const renderFooter = () => {
|
|
return screen === PASSWORD_PROMPT_SCREEN || !completedLongPress
|
|
? renderPasswordPromptFooter()
|
|
: renderRevealSeedFooter();
|
|
};
|
|
|
|
return (
|
|
<Box
|
|
className="page-container"
|
|
paddingTop={8}
|
|
paddingBottom={8}
|
|
paddingLeft={4}
|
|
paddingRight={4}
|
|
gap={4}
|
|
>
|
|
<Text variant={TextVariant.headingLg}>{t('secretRecoveryPhrase')}</Text>
|
|
<Text variant={TextVariant.bodyMd}>
|
|
{t('revealSeedWordsDescription1', [
|
|
<Button
|
|
key="srp-learn-srp"
|
|
variant={BUTTON_VARIANT.LINK}
|
|
size={BUTTON_SIZES.INHERIT}
|
|
as="a"
|
|
href={ZENDESK_URLS.SECRET_RECOVERY_PHRASE}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
{t('revealSeedWordsSRPName')}
|
|
</Button>,
|
|
<Text
|
|
key="reveal-seed-word-part-3"
|
|
variant={TextVariant.bodyMdBold}
|
|
as="strong"
|
|
>
|
|
{t('revealSeedWordsDescription3')}
|
|
</Text>,
|
|
])}
|
|
</Text>
|
|
<Text variant={TextVariant.bodyMd}>
|
|
{t('revealSeedWordsDescription2', [
|
|
<Button
|
|
key="srp-learn-more-non-custodial"
|
|
variant={BUTTON_VARIANT.LINK}
|
|
size={BUTTON_SIZES.INHERIT}
|
|
as="a"
|
|
href={ZENDESK_URLS.NON_CUSTODIAL_WALLET}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
{t('revealSeedWordsNonCustodialWallet')}
|
|
</Button>,
|
|
])}
|
|
</Text>
|
|
{renderWarning()}
|
|
{renderContent()}
|
|
{renderFooter()}
|
|
</Box>
|
|
);
|
|
};
|
|
|
|
export default RevealSeedPage;
|