mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
d8c685ba2e
The filename is seeded by a simple use of Math.random() pulling from an alphanumeric character bank, as opposed to a more cryptographically random solution. This provides a simple layer of difficulty for bad actors to seek out the recovery phrase file.
182 lines
5.0 KiB
JavaScript
182 lines
5.0 KiB
JavaScript
import React, { Component } from 'react'
|
|
import { connect } from 'react-redux'
|
|
import PropTypes from 'prop-types'
|
|
import classnames from 'classnames'
|
|
import { requestRevealSeedWords } from '../../store/actions'
|
|
import { DEFAULT_ROUTE } from '../../helpers/constants/routes'
|
|
import ExportTextContainer from '../../components/ui/export-text-container'
|
|
|
|
import Button from '../../components/ui/button'
|
|
|
|
const PASSWORD_PROMPT_SCREEN = 'PASSWORD_PROMPT_SCREEN'
|
|
const REVEAL_SEED_SCREEN = 'REVEAL_SEED_SCREEN'
|
|
|
|
class RevealSeedPage extends Component {
|
|
state = {
|
|
screen: PASSWORD_PROMPT_SCREEN,
|
|
password: '',
|
|
seedWords: null,
|
|
error: null,
|
|
}
|
|
|
|
componentDidMount () {
|
|
const passwordBox = document.getElementById('password-box')
|
|
if (passwordBox) {
|
|
passwordBox.focus()
|
|
}
|
|
}
|
|
|
|
handleSubmit (event) {
|
|
event.preventDefault()
|
|
this.setState({ seedWords: null, error: null })
|
|
this.props.requestRevealSeedWords(this.state.password)
|
|
.then(seedWords => this.setState({ seedWords, screen: REVEAL_SEED_SCREEN }))
|
|
.catch(error => this.setState({ error: error.message }))
|
|
}
|
|
|
|
renderWarning () {
|
|
return (
|
|
<div className="page-container__warning-container">
|
|
<img className="page-container__warning-icon" src="images/warning.svg" alt="" />
|
|
<div className="page-container__warning-message">
|
|
<div className="page-container__warning-title">
|
|
{this.context.t('revealSeedWordsWarningTitle')}
|
|
</div>
|
|
<div>
|
|
{this.context.t('revealSeedWordsWarning')}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
renderContent () {
|
|
return this.state.screen === PASSWORD_PROMPT_SCREEN
|
|
? this.renderPasswordPromptContent()
|
|
: this.renderRevealSeedContent()
|
|
}
|
|
|
|
renderPasswordPromptContent () {
|
|
const { t } = this.context
|
|
|
|
return (
|
|
<form onSubmit={event => this.handleSubmit(event)}>
|
|
<label className="input-label" htmlFor="password-box">{t('enterPasswordContinue')}</label>
|
|
<div className="input-group">
|
|
<input
|
|
type="password"
|
|
placeholder={t('password')}
|
|
id="password-box"
|
|
value={this.state.password}
|
|
onChange={event => this.setState({ password: event.target.value })}
|
|
className={classnames('form-control', { 'form-control--error': this.state.error })}
|
|
/>
|
|
</div>
|
|
{
|
|
this.state.error && (
|
|
<div className="reveal-seed__error">
|
|
{this.state.error}
|
|
</div>
|
|
)}
|
|
</form>
|
|
)
|
|
}
|
|
|
|
renderRevealSeedContent () {
|
|
const { t } = this.context
|
|
|
|
return (
|
|
<div>
|
|
<label className="reveal-seed__label">{t('yourPrivateSeedPhrase')}</label>
|
|
<ExportTextContainer text={this.state.seedWords} />
|
|
</div>
|
|
)
|
|
}
|
|
|
|
renderFooter () {
|
|
return this.state.screen === PASSWORD_PROMPT_SCREEN
|
|
? this.renderPasswordPromptFooter()
|
|
: this.renderRevealSeedFooter()
|
|
}
|
|
|
|
renderPasswordPromptFooter () {
|
|
return (
|
|
<div className="page-container__footer">
|
|
<header>
|
|
<Button
|
|
type="default"
|
|
large
|
|
className="page-container__footer-button"
|
|
onClick={() => this.props.history.push(DEFAULT_ROUTE)}
|
|
>
|
|
{this.context.t('cancel')}
|
|
</Button>
|
|
<Button
|
|
type="secondary"
|
|
large
|
|
className="page-container__footer-button"
|
|
onClick={event => this.handleSubmit(event)}
|
|
disabled={this.state.password === ''}
|
|
>
|
|
{this.context.t('next')}
|
|
</Button>
|
|
</header>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
renderRevealSeedFooter () {
|
|
return (
|
|
<div className="page-container__footer">
|
|
<Button
|
|
type="default"
|
|
large
|
|
className="page-container__footer-button"
|
|
onClick={() => this.props.history.push(DEFAULT_ROUTE)}
|
|
>
|
|
{this.context.t('close')}
|
|
</Button>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
render () {
|
|
return (
|
|
<div className="page-container">
|
|
<div className="page-container__header">
|
|
<div className="page-container__title">
|
|
{this.context.t('revealSeedWordsTitle')}
|
|
</div>
|
|
<div className="page-container__subtitle">
|
|
{this.context.t('revealSeedWordsDescription')}
|
|
</div>
|
|
</div>
|
|
<div className="page-container__content">
|
|
{this.renderWarning()}
|
|
<div className="reveal-seed__content">
|
|
{this.renderContent()}
|
|
</div>
|
|
</div>
|
|
{this.renderFooter()}
|
|
</div>
|
|
)
|
|
}
|
|
}
|
|
|
|
RevealSeedPage.propTypes = {
|
|
requestRevealSeedWords: PropTypes.func,
|
|
history: PropTypes.object,
|
|
}
|
|
|
|
RevealSeedPage.contextTypes = {
|
|
t: PropTypes.func,
|
|
}
|
|
|
|
const mapDispatchToProps = dispatch => {
|
|
return {
|
|
requestRevealSeedWords: password => dispatch(requestRevealSeedWords(password)),
|
|
}
|
|
}
|
|
|
|
export default connect(null, mapDispatchToProps)(RevealSeedPage)
|