1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-23 09:52:26 +01:00
metamask-extension/ui/app/components/pages/signature-request.js
2018-03-28 13:23:50 -07:00

285 lines
6.9 KiB
JavaScript

const { Component } = require('react')
const h = require('react-hyperscript')
const PropTypes = require('prop-types')
const Identicon = require('../identicon')
const { connect } = require('react-redux')
const ethUtil = require('ethereumjs-util')
const classnames = require('classnames')
const AccountDropdownMini = require('../dropdowns/account-dropdown-mini')
const t = require('../../../i18n')
const { conversionUtil } = require('../../conversion-util')
const { DEFAULT_ROUTE } = require('../../routes')
const {
getSelectedAccount,
getCurrentAccountWithSendEtherInfo,
getSelectedAddress,
accountsWithSendEtherInfoSelector,
conversionRateSelector,
} = require('../../selectors.js')
class SignatureRequest extends Component {
constructor (props) {
super(props)
this.state = {
selectedAccount: props.selectedAccount,
accountDropdownOpen: false,
}
}
componentWillMount () {
const {
unapprovedMsgCount = 0,
unapprovedPersonalMsgCount = 0,
unapprovedTypedMessagesCount = 0,
} = this.props
if (unapprovedMsgCount + unapprovedPersonalMsgCount + unapprovedTypedMessagesCount < 1) {
this.props.history.push(DEFAULT_ROUTE)
}
}
renderHeader () {
return h('div.request-signature__header', [
h('div.request-signature__header-background'),
h('div.request-signature__header__text', t('sigRequest')),
h('div.request-signature__header__tip-container', [
h('div.request-signature__header__tip'),
]),
])
}
renderAccountDropdown () {
const {
selectedAccount,
accountDropdownOpen,
} = this.state
const { accounts } = this.props
return h('div.request-signature__account', [
h('div.request-signature__account-text', [t('account') + ':']),
h(AccountDropdownMini, {
selectedAccount,
accounts,
onSelect: selectedAccount => this.setState({ selectedAccount }),
dropdownOpen: accountDropdownOpen,
openDropdown: () => this.setState({ accountDropdownOpen: true }),
closeDropdown: () => this.setState({ accountDropdownOpen: false }),
}),
])
}
renderBalance () {
const { balance, conversionRate } = this.props
const balanceInEther = conversionUtil(balance, {
fromNumericBase: 'hex',
toNumericBase: 'dec',
fromDenomination: 'WEI',
numberOfDecimals: 6,
conversionRate,
})
return h('div.request-signature__balance', [
h('div.request-signature__balance-text', [t('balance')]),
h('div.request-signature__balance-value', `${balanceInEther} ETH`),
])
}
renderAccountInfo () {
return h('div.request-signature__account-info', [
this.renderAccountDropdown(),
this.renderRequestIcon(),
this.renderBalance(),
])
}
renderRequestIcon () {
const { requesterAddress } = this.props
return h('div.request-signature__request-icon', [
h(Identicon, {
diameter: 40,
address: requesterAddress,
}),
])
}
renderRequestInfo () {
return h('div.request-signature__request-info', [
h('div.request-signature__headline', [
t('yourSigRequested'),
]),
])
}
msgHexToText (hex) {
try {
const stripped = ethUtil.stripHexPrefix(hex)
const buff = Buffer.from(stripped, 'hex')
return buff.toString('utf8')
} catch (e) {
return hex
}
}
renderBody () {
let rows
let notice = t('youSign') + ':'
const { txData } = this.props
const { type, msgParams: { data } } = txData
if (type === 'personal_sign') {
rows = [{ name: t('message'), value: this.msgHexToText(data) }]
} else if (type === 'eth_signTypedData') {
rows = data
} else if (type === 'eth_sign') {
rows = [{ name: t('message'), value: data }]
notice = t('signNotice')
}
return h('div.request-signature__body', {}, [
this.renderAccountInfo(),
this.renderRequestInfo(),
h('div.request-signature__notice', {
className: classnames({
'request-signature__notice': type === 'personal_sign' || type === 'eth_signTypedData',
'request-signature__warning': type === 'eth_sign',
}),
}, [notice]),
h('div.request-signature__rows', [
...rows.map(({ name, value }) => {
return h('div.request-signature__row', [
h('div.request-signature__row-title', [`${name}:`]),
h('div.request-signature__row-value', value),
])
}),
]),
])
}
renderFooter () {
const {
txData = {},
signPersonalMessage,
signTypedMessage,
cancelPersonalMessage,
cancelTypedMessage,
signMessage,
cancelMessage,
history,
} = this.props
const { type } = txData
let cancel = () => Promise.resolve()
let sign = () => Promise.resolve()
const { msgParams: params = {}, id } = txData
params.id = id
params.metamaskId = id
switch (type) {
case 'personal_sign':
cancel = () => cancelPersonalMessage(params)
sign = () => signPersonalMessage(params)
break
case 'eth_signTypedData':
cancel = () => cancelTypedMessage(params)
sign = () => signTypedMessage(params)
break
case 'eth_sign':
cancel = () => cancelMessage(params)
sign = () => signMessage(params)
break
}
return h('div.request-signature__footer', [
h('button.btn-secondary--lg.request-signature__footer__cancel-button', {
onClick: () => {
cancel().then(() => history.push(DEFAULT_ROUTE))
},
}, t('cancel')),
h('button.btn-primary--lg', {
onClick: () => {
sign().then(() => history.push(DEFAULT_ROUTE))
},
}, t('sign')),
])
}
render () {
return (
h('div.request-signature__container', [
this.renderHeader(),
this.renderBody(),
this.renderFooter(),
])
)
}
}
SignatureRequest.propTypes = {
txData: PropTypes.object,
signPersonalMessage: PropTypes.func,
cancelPersonalMessage: PropTypes.func,
signTypedMessage: PropTypes.func,
cancelTypedMessage: PropTypes.func,
signMessage: PropTypes.func,
cancelMessage: PropTypes.func,
requesterAddress: PropTypes.string,
accounts: PropTypes.array,
conversionRate: PropTypes.number,
balance: PropTypes.string,
selectedAccount: PropTypes.object,
history: PropTypes.object,
unapprovedMsgCount: PropTypes.number,
unapprovedPersonalMsgCount: PropTypes.number,
unapprovedTypedMessagesCount: PropTypes.number,
}
const mapStateToProps = state => {
return {
balance: getSelectedAccount(state).balance,
selectedAccount: getCurrentAccountWithSendEtherInfo(state),
selectedAddress: getSelectedAddress(state),
requester: null,
requesterAddress: null,
accounts: accountsWithSendEtherInfoSelector(state),
conversionRate: conversionRateSelector(state),
}
}
module.exports = connect(mapStateToProps)(SignatureRequest)