import { ethers } from 'ethers'; import React, { PureComponent } from 'react'; import PropTypes from 'prop-types'; import TextField from '../../../../components/ui/text-field'; import Button from '../../../../components/ui/button'; import { INITIALIZE_SELECT_ACTION_ROUTE, INITIALIZE_END_OF_FLOW_ROUTE, } from '../../../../helpers/constants/routes'; const { isValidMnemonic } = ethers.utils; export default class ImportWithSeedPhrase extends PureComponent { static contextTypes = { t: PropTypes.func, metricsEvent: PropTypes.func, }; static propTypes = { history: PropTypes.object, onSubmit: PropTypes.func.isRequired, setSeedPhraseBackedUp: PropTypes.func, initializeThreeBox: PropTypes.func, }; state = { seedPhrase: '', showSeedPhrase: false, password: '', confirmPassword: '', seedPhraseError: '', passwordError: '', confirmPasswordError: '', termsChecked: false, }; parseSeedPhrase = (seedPhrase) => (seedPhrase || '').trim().toLowerCase().match(/\w+/gu)?.join(' ') || ''; UNSAFE_componentWillMount() { this._onBeforeUnload = () => this.context.metricsEvent({ eventOpts: { category: 'Onboarding', action: 'Import Seed Phrase', name: 'Close window on import screen', }, customVariables: { errorLabel: 'Seed Phrase Error', errorMessage: this.state.seedPhraseError, }, }); window.addEventListener('beforeunload', this._onBeforeUnload); } componentWillUnmount() { window.removeEventListener('beforeunload', this._onBeforeUnload); } handleSeedPhraseChange(seedPhrase) { let seedPhraseError = ''; if (seedPhrase) { const parsedSeedPhrase = this.parseSeedPhrase(seedPhrase); const wordCount = parsedSeedPhrase.split(/\s/u).length; if (wordCount % 3 !== 0 || wordCount > 24 || wordCount < 12) { seedPhraseError = this.context.t('seedPhraseReq'); } else if (!isValidMnemonic(parsedSeedPhrase)) { seedPhraseError = this.context.t('invalidSeedPhrase'); } } this.setState({ seedPhrase, seedPhraseError }); } handlePasswordChange(password) { const { t } = this.context; this.setState((state) => { const { confirmPassword } = state; let confirmPasswordError = ''; let passwordError = ''; if (password && password.length < 8) { passwordError = t('passwordNotLongEnough'); } if (confirmPassword && password !== confirmPassword) { confirmPasswordError = t('passwordsDontMatch'); } return { password, passwordError, confirmPasswordError, }; }); } handleConfirmPasswordChange(confirmPassword) { const { t } = this.context; this.setState((state) => { const { password } = state; let confirmPasswordError = ''; if (password !== confirmPassword) { confirmPasswordError = t('passwordsDontMatch'); } return { confirmPassword, confirmPasswordError, }; }); } handleImport = async (event) => { event.preventDefault(); if (!this.isValid()) { return; } const { password, seedPhrase } = this.state; const { history, onSubmit, setSeedPhraseBackedUp, initializeThreeBox, } = this.props; try { await onSubmit(password, this.parseSeedPhrase(seedPhrase)); this.context.metricsEvent({ eventOpts: { category: 'Onboarding', action: 'Import Seed Phrase', name: 'Import Complete', }, }); setSeedPhraseBackedUp(true).then(async () => { initializeThreeBox(); history.replace(INITIALIZE_END_OF_FLOW_ROUTE); }); } catch (error) { this.setState({ seedPhraseError: error.message }); } }; isValid() { const { seedPhrase, password, confirmPassword, passwordError, confirmPasswordError, seedPhraseError, } = this.state; if ( !password || !confirmPassword || !seedPhrase || password !== confirmPassword ) { return false; } if (password.length < 8) { return false; } return !passwordError && !confirmPasswordError && !seedPhraseError; } onTermsKeyPress = ({ key }) => { if (key === ' ' || key === 'Enter') { this.toggleTermsCheck(); } }; toggleTermsCheck = () => { this.context.metricsEvent({ eventOpts: { category: 'Onboarding', action: 'Import Seed Phrase', name: 'Check ToS', }, }); this.setState((prevState) => ({ termsChecked: !prevState.termsChecked, })); }; toggleShowSeedPhrase = () => { this.setState(({ showSeedPhrase }) => ({ showSeedPhrase: !showSeedPhrase, })); }; render() { const { t } = this.context; const { seedPhraseError, showSeedPhrase, passwordError, confirmPasswordError, termsChecked, } = this.state; return (
{ e.preventDefault(); this.context.metricsEvent({ eventOpts: { category: 'Onboarding', action: 'Import Seed Phrase', name: 'Go Back from Onboarding Import', }, customVariables: { errorLabel: 'Seed Phrase Error', errorMessage: seedPhraseError, }, }); this.props.history.push(INITIALIZE_SELECT_ACTION_ROUTE); }} href="#" > {`< ${t('back')}`}
{t('importAccountSeedPhrase')}
{t('secretPhrase')}
{showSeedPhrase ? (