Merge pull request #4190 from MetaMask/i3614-unlock
Add new unlock screen designfeature/default_network_editable
commit
6bd1b21d3b
@ -0,0 +1,106 @@ |
||||
import React, { Component } from 'react' |
||||
import PropTypes from 'prop-types' |
||||
import classnames from 'classnames' |
||||
|
||||
const { ENVIRONMENT_TYPE_NOTIFICATION } = require('../../../../app/scripts/lib/enums') |
||||
const { DEFAULT_ROUTE, CONFIRM_TRANSACTION_ROUTE } = require('../../routes') |
||||
const Identicon = require('../identicon') |
||||
const NetworkIndicator = require('../network') |
||||
|
||||
class AppHeader extends Component { |
||||
static propTypes = { |
||||
history: PropTypes.object, |
||||
location: PropTypes.object, |
||||
network: PropTypes.string, |
||||
provider: PropTypes.object, |
||||
networkDropdownOpen: PropTypes.bool, |
||||
showNetworkDropdown: PropTypes.func, |
||||
hideNetworkDropdown: PropTypes.func, |
||||
toggleAccountMenu: PropTypes.func, |
||||
selectedAddress: PropTypes.string, |
||||
isUnlocked: PropTypes.bool, |
||||
} |
||||
|
||||
static contextTypes = { |
||||
t: PropTypes.func, |
||||
} |
||||
|
||||
handleNetworkIndicatorClick (event) { |
||||
event.preventDefault() |
||||
event.stopPropagation() |
||||
|
||||
const { networkDropdownOpen, showNetworkDropdown, hideNetworkDropdown } = this.props |
||||
|
||||
return networkDropdownOpen === false |
||||
? showNetworkDropdown() |
||||
: hideNetworkDropdown() |
||||
} |
||||
|
||||
renderAccountMenu () { |
||||
const { isUnlocked, toggleAccountMenu, selectedAddress } = this.props |
||||
|
||||
return isUnlocked && ( |
||||
<div |
||||
className="account-menu__icon" |
||||
onClick={toggleAccountMenu} |
||||
> |
||||
<Identicon |
||||
address={selectedAddress} |
||||
diameter={32} |
||||
/> |
||||
</div> |
||||
) |
||||
} |
||||
|
||||
render () { |
||||
const { |
||||
network, |
||||
provider, |
||||
history, |
||||
location, |
||||
isUnlocked, |
||||
} = this.props |
||||
|
||||
if (window.METAMASK_UI_TYPE === ENVIRONMENT_TYPE_NOTIFICATION) { |
||||
return null |
||||
} |
||||
|
||||
return ( |
||||
<div |
||||
className={classnames('app-header', { 'app-header--back-drop': isUnlocked })}> |
||||
<div className="app-header__contents"> |
||||
<div |
||||
className="app-header__logo-container" |
||||
onClick={() => history.push(DEFAULT_ROUTE)} |
||||
> |
||||
<img |
||||
className="app-header__metafox" |
||||
src="/images/metamask-fox.svg" |
||||
height={42} |
||||
width={42} |
||||
/> |
||||
<div className="flex-row"> |
||||
<h1>{ this.context.t('appName') }</h1> |
||||
<div className="app-header__beta-label"> |
||||
{ this.context.t('beta') } |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div className="app-header__account-menu-container"> |
||||
<div className="network-component-wrapper"> |
||||
<NetworkIndicator |
||||
network={network} |
||||
provider={provider} |
||||
onClick={event => this.handleNetworkIndicatorClick(event)} |
||||
disabled={location.pathname === CONFIRM_TRANSACTION_ROUTE} |
||||
/> |
||||
</div> |
||||
{ this.renderAccountMenu() } |
||||
</div> |
||||
</div> |
||||
</div> |
||||
) |
||||
} |
||||
} |
||||
|
||||
export default AppHeader |
@ -0,0 +1,38 @@ |
||||
import { connect } from 'react-redux' |
||||
import { withRouter } from 'react-router-dom' |
||||
import { compose } from 'recompose' |
||||
|
||||
import AppHeader from './app-header.component' |
||||
const actions = require('../../actions') |
||||
|
||||
const mapStateToProps = state => { |
||||
const { appState, metamask } = state |
||||
const { networkDropdownOpen } = appState |
||||
const { |
||||
network, |
||||
provider, |
||||
selectedAddress, |
||||
isUnlocked, |
||||
} = metamask |
||||
|
||||
return { |
||||
networkDropdownOpen, |
||||
network, |
||||
provider, |
||||
selectedAddress, |
||||
isUnlocked, |
||||
} |
||||
} |
||||
|
||||
const mapDispatchToProps = dispatch => { |
||||
return { |
||||
showNetworkDropdown: () => dispatch(actions.showNetworkDropdown()), |
||||
hideNetworkDropdown: () => dispatch(actions.hideNetworkDropdown()), |
||||
toggleAccountMenu: () => dispatch(actions.toggleAccountMenu()), |
||||
} |
||||
} |
||||
|
||||
export default compose( |
||||
withRouter, |
||||
connect(mapStateToProps, mapDispatchToProps) |
||||
)(AppHeader) |
@ -0,0 +1,2 @@ |
||||
import AppHeader from './app-header.container' |
||||
module.exports = AppHeader |
@ -0,0 +1,2 @@ |
||||
import UnlockPage from './unlock-page.container' |
||||
module.exports = UnlockPage |
@ -0,0 +1,181 @@ |
||||
import React, { Component } from 'react' |
||||
import PropTypes from 'prop-types' |
||||
import Button from 'material-ui/Button' |
||||
import TextField from '../../text-field' |
||||
|
||||
const { ENVIRONMENT_TYPE_POPUP } = require('../../../../../app/scripts/lib/enums') |
||||
const { getEnvironmentType } = require('../../../../../app/scripts/lib/util') |
||||
const getCaretCoordinates = require('textarea-caret') |
||||
const EventEmitter = require('events').EventEmitter |
||||
const Mascot = require('../../mascot') |
||||
const { DEFAULT_ROUTE, RESTORE_VAULT_ROUTE } = require('../../../routes') |
||||
|
||||
class UnlockPage extends Component { |
||||
static contextTypes = { |
||||
t: PropTypes.func, |
||||
} |
||||
|
||||
constructor (props) { |
||||
super(props) |
||||
|
||||
this.state = { |
||||
password: '', |
||||
error: null, |
||||
} |
||||
|
||||
this.animationEventEmitter = new EventEmitter() |
||||
} |
||||
|
||||
componentWillMount () { |
||||
const { isUnlocked, history } = this.props |
||||
|
||||
if (isUnlocked) { |
||||
history.push(DEFAULT_ROUTE) |
||||
} |
||||
} |
||||
|
||||
tryUnlockMetamask (password) { |
||||
const { tryUnlockMetamask, history } = this.props |
||||
tryUnlockMetamask(password) |
||||
.then(() => history.push(DEFAULT_ROUTE)) |
||||
.catch(({ message }) => this.setState({ error: message })) |
||||
} |
||||
|
||||
handleSubmit (event) { |
||||
event.preventDefault() |
||||
event.stopPropagation() |
||||
|
||||
const { password } = this.state |
||||
const { tryUnlockMetamask, history } = this.props |
||||
|
||||
if (password === '') { |
||||
return |
||||
} |
||||
|
||||
this.setState({ error: null }) |
||||
|
||||
tryUnlockMetamask(password) |
||||
.then(() => history.push(DEFAULT_ROUTE)) |
||||
.catch(({ message }) => this.setState({ error: message })) |
||||
} |
||||
|
||||
handleInputChange ({ target }) { |
||||
this.setState({ password: target.value, error: null }) |
||||
|
||||
// tell mascot to look at page action
|
||||
const element = target |
||||
const boundingRect = element.getBoundingClientRect() |
||||
const coordinates = getCaretCoordinates(element, element.selectionEnd) |
||||
this.animationEventEmitter.emit('point', { |
||||
x: boundingRect.left + coordinates.left - element.scrollLeft, |
||||
y: boundingRect.top + coordinates.top - element.scrollTop, |
||||
}) |
||||
} |
||||
|
||||
renderSubmitButton () { |
||||
const style = { |
||||
backgroundColor: '#f7861c', |
||||
color: 'white', |
||||
marginTop: '20px', |
||||
height: '60px', |
||||
fontWeight: '400', |
||||
boxShadow: 'none', |
||||
borderRadius: '4px', |
||||
} |
||||
|
||||
return ( |
||||
<Button |
||||
type="submit" |
||||
style={style} |
||||
disabled={!this.state.password} |
||||
fullWidth |
||||
variant="raised" |
||||
size="large" |
||||
onClick={event => this.handleSubmit(event)} |
||||
disableRipple |
||||
> |
||||
{ this.context.t('login') } |
||||
</Button> |
||||
) |
||||
} |
||||
|
||||
render () { |
||||
const { error } = this.state |
||||
|
||||
return ( |
||||
<div className="unlock-page__container"> |
||||
<div className="unlock-page"> |
||||
<div className="unlock-page__mascot-container"> |
||||
<Mascot |
||||
animationEventEmitter={this.animationEventEmitter} |
||||
width="120" |
||||
height="120" |
||||
/> |
||||
</div> |
||||
<h1 className="unlock-page__title"> |
||||
{ this.context.t('welcomeBack') } |
||||
</h1> |
||||
<div>{ this.context.t('unlockMessage') }</div> |
||||
<form |
||||
className="unlock-page__form" |
||||
onSubmit={event => this.handleSubmit(event)} |
||||
> |
||||
<TextField |
||||
id="password" |
||||
label="Password" |
||||
type="password" |
||||
value={this.state.password} |
||||
onChange={event => this.handleInputChange(event)} |
||||
error={error} |
||||
autoFocus |
||||
autoComplete="current-password" |
||||
fullWidth |
||||
/> |
||||
</form> |
||||
{ this.renderSubmitButton() } |
||||
<div className="unlock-page__links"> |
||||
<div |
||||
className="unlock-page__link" |
||||
onClick={() => { |
||||
this.props.markPasswordForgotten() |
||||
this.props.history.push(RESTORE_VAULT_ROUTE) |
||||
|
||||
if (getEnvironmentType(window.location.href) === ENVIRONMENT_TYPE_POPUP) { |
||||
global.platform.openExtensionInBrowser() |
||||
} |
||||
}} |
||||
> |
||||
{ this.context.t('restoreFromSeed') } |
||||
</div> |
||||
<div |
||||
className="unlock-page__link unlock-page__link--import" |
||||
onClick={() => { |
||||
this.props.markPasswordForgotten() |
||||
this.props.history.push(RESTORE_VAULT_ROUTE) |
||||
|
||||
if (getEnvironmentType(window.location.href) === ENVIRONMENT_TYPE_POPUP) { |
||||
global.platform.openExtensionInBrowser() |
||||
} |
||||
}} |
||||
> |
||||
{ this.context.t('importUsingSeed') } |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
) |
||||
} |
||||
} |
||||
|
||||
UnlockPage.propTypes = { |
||||
forgotPassword: PropTypes.func, |
||||
tryUnlockMetamask: PropTypes.func, |
||||
markPasswordForgotten: PropTypes.func, |
||||
history: PropTypes.object, |
||||
isUnlocked: PropTypes.bool, |
||||
t: PropTypes.func, |
||||
useOldInterface: PropTypes.func, |
||||
setNetworkEndpoints: PropTypes.func, |
||||
} |
||||
|
||||
export default UnlockPage |
@ -0,0 +1,33 @@ |
||||
import { connect } from 'react-redux' |
||||
import { withRouter } from 'react-router-dom' |
||||
import { compose } from 'recompose' |
||||
|
||||
const { |
||||
tryUnlockMetamask, |
||||
forgotPassword, |
||||
markPasswordForgotten, |
||||
setNetworkEndpoints, |
||||
} = require('../../../actions') |
||||
|
||||
import UnlockPage from './unlock-page.component' |
||||
|
||||
const mapStateToProps = state => { |
||||
const { metamask: { isUnlocked } } = state |
||||
return { |
||||
isUnlocked, |
||||
} |
||||
} |
||||
|
||||
const mapDispatchToProps = dispatch => { |
||||
return { |
||||
forgotPassword: () => dispatch(forgotPassword()), |
||||
tryUnlockMetamask: password => dispatch(tryUnlockMetamask(password)), |
||||
markPasswordForgotten: () => dispatch(markPasswordForgotten()), |
||||
setNetworkEndpoints: type => dispatch(setNetworkEndpoints(type)), |
||||
} |
||||
} |
||||
|
||||
export default compose( |
||||
withRouter, |
||||
connect(mapStateToProps, mapDispatchToProps) |
||||
)(UnlockPage) |
@ -0,0 +1,51 @@ |
||||
.unlock-page { |
||||
display: flex; |
||||
flex-direction: column; |
||||
justify-content: flex-start; |
||||
align-items: center; |
||||
width: 357px; |
||||
padding: 30px; |
||||
font-weight: 400; |
||||
color: $silver-chalice; |
||||
|
||||
&__container { |
||||
background: $white; |
||||
display: flex; |
||||
align-self: stretch; |
||||
justify-content: center; |
||||
flex: 1 0 auto; |
||||
} |
||||
|
||||
&__mascot-container { |
||||
margin-top: 24px; |
||||
} |
||||
|
||||
&__title { |
||||
margin-top: 5px; |
||||
font-size: 2rem; |
||||
font-weight: 800; |
||||
color: $tundora; |
||||
} |
||||
|
||||
&__form { |
||||
width: 100%; |
||||
margin: 56px 0 8px; |
||||
} |
||||
|
||||
&__links { |
||||
margin-top: 25px; |
||||
width: 100%; |
||||
} |
||||
|
||||
&__link { |
||||
cursor: pointer; |
||||
|
||||
&--import { |
||||
color: $ecstasy; |
||||
} |
||||
|
||||
&--use-classic { |
||||
margin-top: 10px; |
||||
} |
||||
} |
||||
} |
@ -1,194 +0,0 @@ |
||||
const { Component } = require('react') |
||||
const PropTypes = require('prop-types') |
||||
const connect = require('../../metamask-connect') |
||||
const h = require('react-hyperscript') |
||||
const { withRouter } = require('react-router-dom') |
||||
const { compose } = require('recompose') |
||||
const { |
||||
tryUnlockMetamask, |
||||
forgotPassword, |
||||
markPasswordForgotten, |
||||
setNetworkEndpoints, |
||||
setFeatureFlag, |
||||
} = require('../../actions') |
||||
const { ENVIRONMENT_TYPE_POPUP } = require('../../../../app/scripts/lib/enums') |
||||
const { getEnvironmentType } = require('../../../../app/scripts/lib/util') |
||||
const getCaretCoordinates = require('textarea-caret') |
||||
const EventEmitter = require('events').EventEmitter |
||||
const Mascot = require('../mascot') |
||||
const { OLD_UI_NETWORK_TYPE } = require('../../../../app/scripts/controllers/network/enums') |
||||
const { DEFAULT_ROUTE, RESTORE_VAULT_ROUTE } = require('../../routes') |
||||
|
||||
class UnlockScreen extends Component { |
||||
constructor (props) { |
||||
super(props) |
||||
|
||||
this.state = { |
||||
error: null, |
||||
} |
||||
|
||||
this.animationEventEmitter = new EventEmitter() |
||||
} |
||||
|
||||
componentWillMount () { |
||||
const { isUnlocked, history } = this.props |
||||
|
||||
if (isUnlocked) { |
||||
history.push(DEFAULT_ROUTE) |
||||
} |
||||
} |
||||
|
||||
componentDidMount () { |
||||
const passwordBox = document.getElementById('password-box') |
||||
|
||||
if (passwordBox) { |
||||
passwordBox.focus() |
||||
} |
||||
} |
||||
|
||||
tryUnlockMetamask (password) { |
||||
const { tryUnlockMetamask, history } = this.props |
||||
tryUnlockMetamask(password) |
||||
.then(() => history.push(DEFAULT_ROUTE)) |
||||
.catch(({ message }) => this.setState({ error: message })) |
||||
} |
||||
|
||||
onSubmit (event) { |
||||
const input = document.getElementById('password-box') |
||||
const password = input.value |
||||
this.tryUnlockMetamask(password) |
||||
} |
||||
|
||||
onKeyPress (event) { |
||||
if (event.key === 'Enter') { |
||||
this.submitPassword(event) |
||||
} |
||||
} |
||||
|
||||
submitPassword (event) { |
||||
var element = event.target |
||||
var password = element.value |
||||
// reset input
|
||||
element.value = '' |
||||
this.tryUnlockMetamask(password) |
||||
} |
||||
|
||||
inputChanged (event) { |
||||
// tell mascot to look at page action
|
||||
var element = event.target |
||||
var boundingRect = element.getBoundingClientRect() |
||||
var coordinates = getCaretCoordinates(element, element.selectionEnd) |
||||
this.animationEventEmitter.emit('point', { |
||||
x: boundingRect.left + coordinates.left - element.scrollLeft, |
||||
y: boundingRect.top + coordinates.top - element.scrollTop, |
||||
}) |
||||
} |
||||
|
||||
render () { |
||||
const { error } = this.state |
||||
return ( |
||||
h('.unlock-screen', [ |
||||
|
||||
h(Mascot, { |
||||
animationEventEmitter: this.animationEventEmitter, |
||||
}), |
||||
|
||||
h('h1', { |
||||
style: { |
||||
fontSize: '1.4em', |
||||
textTransform: 'uppercase', |
||||
color: '#7F8082', |
||||
}, |
||||
}, this.props.t('appName')), |
||||
|
||||
h('input.large-input', { |
||||
type: 'password', |
||||
id: 'password-box', |
||||
placeholder: 'enter password', |
||||
style: { |
||||
background: 'white', |
||||
}, |
||||
onKeyPress: this.onKeyPress.bind(this), |
||||
onInput: this.inputChanged.bind(this), |
||||
}), |
||||
|
||||
h('.error', { |
||||
style: { |
||||
display: error ? 'block' : 'none', |
||||
padding: '0 20px', |
||||
textAlign: 'center', |
||||
}, |
||||
}, error), |
||||
|
||||
h('button.primary.cursor-pointer', { |
||||
onClick: this.onSubmit.bind(this), |
||||
style: { |
||||
margin: 10, |
||||
}, |
||||
}, this.props.t('login')), |
||||
|
||||
h('p.pointer', { |
||||
onClick: () => { |
||||
this.props.markPasswordForgotten() |
||||
this.props.history.push(RESTORE_VAULT_ROUTE) |
||||
|
||||
if (getEnvironmentType(window.location.href) === ENVIRONMENT_TYPE_POPUP) { |
||||
global.platform.openExtensionInBrowser() |
||||
} |
||||
}, |
||||
style: { |
||||
fontSize: '0.8em', |
||||
color: 'rgb(247, 134, 28)', |
||||
textDecoration: 'underline', |
||||
}, |
||||
}, this.props.t('restoreFromSeed')), |
||||
|
||||
h('p.pointer', { |
||||
onClick: () => { |
||||
this.props.useOldInterface() |
||||
.then(() => this.props.setNetworkEndpoints(OLD_UI_NETWORK_TYPE)) |
||||
}, |
||||
style: { |
||||
fontSize: '0.8em', |
||||
color: '#aeaeae', |
||||
textDecoration: 'underline', |
||||
marginTop: '32px', |
||||
}, |
||||
}, this.props.t('classicInterface')), |
||||
]) |
||||
) |
||||
} |
||||
} |
||||
|
||||
UnlockScreen.propTypes = { |
||||
forgotPassword: PropTypes.func, |
||||
tryUnlockMetamask: PropTypes.func, |
||||
markPasswordForgotten: PropTypes.func, |
||||
history: PropTypes.object, |
||||
isUnlocked: PropTypes.bool, |
||||
t: PropTypes.func, |
||||
useOldInterface: PropTypes.func, |
||||
setNetworkEndpoints: PropTypes.func, |
||||
} |
||||
|
||||
const mapStateToProps = state => { |
||||
const { metamask: { isUnlocked } } = state |
||||
return { |
||||
isUnlocked, |
||||
} |
||||
} |
||||
|
||||
const mapDispatchToProps = dispatch => { |
||||
return { |
||||
forgotPassword: () => dispatch(forgotPassword()), |
||||
tryUnlockMetamask: password => dispatch(tryUnlockMetamask(password)), |
||||
markPasswordForgotten: () => dispatch(markPasswordForgotten()), |
||||
useOldInterface: () => dispatch(setFeatureFlag('betaUI', false, 'OLD_UI_NOTIFICATION_MODAL')), |
||||
setNetworkEndpoints: type => dispatch(setNetworkEndpoints(type)), |
||||
} |
||||
} |
||||
|
||||
module.exports = compose( |
||||
withRouter, |
||||
connect(mapStateToProps, mapDispatchToProps) |
||||
)(UnlockScreen) |
@ -0,0 +1,2 @@ |
||||
import TextField from './text-field.component' |
||||
module.exports = TextField |
@ -0,0 +1,54 @@ |
||||
import React from 'react' |
||||
import PropTypes from 'prop-types' |
||||
import { withStyles } from 'material-ui/styles' |
||||
import { default as MaterialTextField } from 'material-ui/TextField' |
||||
|
||||
const styles = { |
||||
cssLabel: { |
||||
'&$cssFocused': { |
||||
color: '#aeaeae', |
||||
}, |
||||
fontWeight: '400', |
||||
color: '#aeaeae', |
||||
}, |
||||
cssFocused: {}, |
||||
cssUnderline: { |
||||
'&:after': { |
||||
backgroundColor: '#f7861c', |
||||
}, |
||||
}, |
||||
} |
||||
|
||||
const TextField = props => { |
||||
const { error, classes, ...textFieldProps } = props |
||||
|
||||
return ( |
||||
<MaterialTextField |
||||
error={Boolean(error)} |
||||
helperText={error} |
||||
InputLabelProps={{ |
||||
FormLabelClasses: { |
||||
root: classes.cssLabel, |
||||
focused: classes.cssFocused, |
||||
}, |
||||
}} |
||||
InputProps={{ |
||||
classes: { |
||||
underline: classes.cssUnderline, |
||||
}, |
||||
}} |
||||
{...textFieldProps} |
||||
/> |
||||
) |
||||
} |
||||
|
||||
TextField.defaultProps = { |
||||
error: null, |
||||
} |
||||
|
||||
TextField.propTypes = { |
||||
error: PropTypes.string, |
||||
classes: PropTypes.object, |
||||
} |
||||
|
||||
export default withStyles(styles)(TextField) |
@ -0,0 +1,24 @@ |
||||
import React from 'react' |
||||
import { storiesOf } from '@storybook/react' |
||||
import TextField from './' |
||||
|
||||
storiesOf('TextField', module) |
||||
.add('text', () => |
||||
<TextField |
||||
label="Text" |
||||
type="text" |
||||
/> |
||||
) |
||||
.add('password', () => |
||||
<TextField |
||||
label="Password" |
||||
type="password" |
||||
/> |
||||
) |
||||
.add('error', () => |
||||
<TextField |
||||
type="text" |
||||
label="Name" |
||||
error="Invalid value" |
||||
/> |
||||
) |
@ -1,3 +1,3 @@ |
||||
@import './unlock.scss'; |
||||
|
||||
@import './reveal-seed.scss'; |
||||
|
||||
@import '../../../../components/pages/unlock-page/unlock-page.scss'; |
||||
|
@ -1,9 +0,0 @@ |
||||
.unlock-page { |
||||
box-shadow: none; |
||||
display: flex; |
||||
flex-direction: column; |
||||
align-items: center; |
||||
justify-content: center; |
||||
background: rgb(247, 247, 247); |
||||
width: 100%; |
||||
} |
@ -1,141 +0,0 @@ |
||||
const inherits = require('util').inherits |
||||
const Component = require('react').Component |
||||
const PropTypes = require('prop-types') |
||||
const h = require('react-hyperscript') |
||||
const connect = require('react-redux').connect |
||||
const actions = require('./actions') |
||||
const getCaretCoordinates = require('textarea-caret') |
||||
const EventEmitter = require('events').EventEmitter |
||||
const { OLD_UI_NETWORK_TYPE } = require('../../app/scripts/controllers/network/enums') |
||||
const { getEnvironmentType } = require('../../app/scripts/lib/util') |
||||
const { ENVIRONMENT_TYPE_POPUP } = require('../../app/scripts/lib/enums') |
||||
|
||||
const Mascot = require('./components/mascot') |
||||
|
||||
UnlockScreen.contextTypes = { |
||||
t: PropTypes.func, |
||||
} |
||||
|
||||
module.exports = connect(mapStateToProps)(UnlockScreen) |
||||
|
||||
|
||||
inherits(UnlockScreen, Component) |
||||
function UnlockScreen () { |
||||
Component.call(this) |
||||
this.animationEventEmitter = new EventEmitter() |
||||
} |
||||
|
||||
function mapStateToProps (state) { |
||||
return { |
||||
warning: state.appState.warning, |
||||
} |
||||
} |
||||
|
||||
UnlockScreen.prototype.render = function () { |
||||
const state = this.props |
||||
const warning = state.warning |
||||
return ( |
||||
h('.unlock-screen', [ |
||||
|
||||
h(Mascot, { |
||||
animationEventEmitter: this.animationEventEmitter, |
||||
}), |
||||
|
||||
h('h1', { |
||||
style: { |
||||
fontSize: '1.4em', |
||||
textTransform: 'uppercase', |
||||
color: '#7F8082', |
||||
}, |
||||
}, this.context.t('appName')), |
||||
|
||||
h('input.large-input', { |
||||
type: 'password', |
||||
id: 'password-box', |
||||
placeholder: 'enter password', |
||||
style: { |
||||
background: 'white', |
||||
}, |
||||
onKeyPress: this.onKeyPress.bind(this), |
||||
onInput: this.inputChanged.bind(this), |
||||
}), |
||||
|
||||
h('.error', { |
||||
style: { |
||||
display: warning ? 'block' : 'none', |
||||
padding: '0 20px', |
||||
textAlign: 'center', |
||||
}, |
||||
}, warning), |
||||
|
||||
h('button.primary.cursor-pointer', { |
||||
onClick: this.onSubmit.bind(this), |
||||
style: { |
||||
margin: 10, |
||||
}, |
||||
}, this.context.t('login')), |
||||
|
||||
h('p.pointer', { |
||||
onClick: () => { |
||||
this.props.dispatch(actions.markPasswordForgotten()) |
||||
if (getEnvironmentType(window.location.href) === ENVIRONMENT_TYPE_POPUP) { |
||||
global.platform.openExtensionInBrowser() |
||||
} |
||||
}, |
||||
style: { |
||||
fontSize: '0.8em', |
||||
color: 'rgb(247, 134, 28)', |
||||
textDecoration: 'underline', |
||||
}, |
||||
}, this.context.t('restoreFromSeed')), |
||||
|
||||
h('p.pointer', { |
||||
onClick: () => { |
||||
this.props.dispatch(actions.setFeatureFlag('betaUI', false, 'OLD_UI_NOTIFICATION_MODAL')) |
||||
.then(() => this.props.dispatch(actions.setNetworkEndpoints(OLD_UI_NETWORK_TYPE))) |
||||
}, |
||||
style: { |
||||
fontSize: '0.8em', |
||||
color: '#aeaeae', |
||||
textDecoration: 'underline', |
||||
marginTop: '32px', |
||||
}, |
||||
}, this.context.t('classicInterface')), |
||||
]) |
||||
) |
||||
} |
||||
|
||||
UnlockScreen.prototype.componentDidMount = function () { |
||||
document.getElementById('password-box').focus() |
||||
} |
||||
|
||||
UnlockScreen.prototype.onSubmit = function (event) { |
||||
const input = document.getElementById('password-box') |
||||
const password = input.value |
||||
this.props.dispatch(actions.tryUnlockMetamask(password)) |
||||
} |
||||
|
||||
UnlockScreen.prototype.onKeyPress = function (event) { |
||||
if (event.key === 'Enter') { |
||||
this.submitPassword(event) |
||||
} |
||||
} |
||||
|
||||
UnlockScreen.prototype.submitPassword = function (event) { |
||||
var element = event.target |
||||
var password = element.value |
||||
// reset input
|
||||
element.value = '' |
||||
this.props.dispatch(actions.tryUnlockMetamask(password)) |
||||
} |
||||
|
||||
UnlockScreen.prototype.inputChanged = function (event) { |
||||
// tell mascot to look at page action
|
||||
var element = event.target |
||||
var boundingRect = element.getBoundingClientRect() |
||||
var coordinates = getCaretCoordinates(element, element.selectionEnd) |
||||
this.animationEventEmitter.emit('point', { |
||||
x: boundingRect.left + coordinates.left - element.scrollLeft, |
||||
y: boundingRect.top + coordinates.top - element.scrollTop, |
||||
}) |
||||
} |
Loading…
Reference in new issue