1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-12 20:57:12 +01:00
metamask-extension/ui/app/components/account-dropdowns.js

326 lines
8.9 KiB
JavaScript
Raw Normal View History

2017-07-18 14:25:16 +02:00
const Component = require('react').Component
const PropTypes = require('prop-types')
2017-07-18 14:25:16 +02:00
const h = require('react-hyperscript')
const actions = require('../actions')
2017-10-14 17:23:44 +02:00
const genAccountLink = require('etherscan-link').createAccountLink
const connect = require('react-redux').connect
2017-07-18 14:25:16 +02:00
const Dropdown = require('./dropdown').Dropdown
const DropdownMenuItem = require('./dropdown').DropdownMenuItem
const copyToClipboard = require('copy-to-clipboard')
const { checksumAddress } = require('../util')
2017-07-18 14:25:16 +02:00
import Identicon from './identicon'
2017-07-18 14:25:16 +02:00
class AccountDropdowns extends Component {
constructor (props) {
super(props)
this.state = {
accountSelectorActive: false,
optionsMenuActive: false,
2017-07-18 14:25:16 +02:00
}
this.accountSelectorToggleClassName = 'accounts-selector'
2017-08-01 22:21:02 +02:00
this.optionsMenuToggleClassName = 'fa-ellipsis-h'
2017-07-18 14:25:16 +02:00
}
renderAccounts () {
2017-08-10 02:40:01 +02:00
const { identities, selected, keyrings } = this.props
2017-07-18 14:25:16 +02:00
return Object.keys(identities).map((key, index) => {
2017-07-18 14:25:16 +02:00
const identity = identities[key]
const isSelected = identity.address === selected
2017-08-10 02:40:01 +02:00
const simpleAddress = identity.address.substring(2).toLowerCase()
const keyring = keyrings.find((kr) => {
return kr.accounts.includes(simpleAddress) ||
kr.accounts.includes(identity.address)
})
2017-07-18 14:25:16 +02:00
return h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => {
this.props.actions.showAccountDetail(identity.address)
},
style: {
marginTop: index === 0 ? '5px' : '',
fontSize: '24px',
2017-08-04 04:17:46 +02:00
},
2017-07-18 14:25:16 +02:00
},
[
h(
Identicon,
{
address: identity.address,
diameter: 32,
style: {
marginLeft: '10px',
},
2017-07-18 14:25:16 +02:00
},
),
this.indicateIfLoose(keyring),
h('span', {
style: {
marginLeft: '20px',
fontSize: '24px',
maxWidth: '145px',
whiteSpace: 'nowrap',
overflow: 'hidden',
textOverflow: 'ellipsis',
2017-08-11 02:49:59 +02:00
},
}, identity.name || ''),
h('span', { style: { marginLeft: '20px', fontSize: '24px' } }, isSelected ? h('.check', '✓') : null),
2017-07-18 14:25:16 +02:00
]
)
})
}
indicateIfLoose (keyring) {
2017-08-10 02:40:01 +02:00
try { // Sometimes keyrings aren't loaded yet:
const type = keyring.type
2017-08-10 02:40:01 +02:00
const isLoose = type !== 'HD Key Tree'
return isLoose ? h('.keyring-label.allcaps', this.context.t('loose')) : null
2017-08-10 02:40:01 +02:00
} catch (e) { return }
}
renderAccountSelector () {
const { actions } = this.props
const { accountSelectorActive } = this.state
return h(
Dropdown,
{
useCssTransition: true, // Hardcoded because account selector is temporarily in app-header
style: {
marginLeft: '-238px',
marginTop: '38px',
minWidth: '180px',
overflowY: 'auto',
maxHeight: '300px',
width: '300px',
},
innerStyle: {
padding: '8px 25px',
},
isOpen: accountSelectorActive,
onClickOutside: (event) => {
const { classList } = event.target
const isNotToggleElement = !classList.contains(this.accountSelectorToggleClassName)
if (accountSelectorActive && isNotToggleElement) {
this.setState({ accountSelectorActive: false })
}
},
},
[
...this.renderAccounts(),
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => actions.addNewAccount(),
},
[
h(
Identicon,
{
style: {
2017-08-04 04:17:46 +02:00
marginLeft: '10px',
},
diameter: 32,
},
),
h('span', { style: { marginLeft: '20px', fontSize: '24px' } }, this.context.t('createAccount')),
],
),
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => actions.showImportPage(),
},
[
h(
Identicon,
{
style: {
2017-08-04 04:17:46 +02:00
marginLeft: '10px',
},
diameter: 32,
},
),
h('span', {
style: {
marginLeft: '20px',
fontSize: '24px',
marginBottom: '5px',
},
}, this.context.t('importAccount')),
]
),
]
)
}
renderAccountOptions () {
const { actions } = this.props
const { optionsMenuActive } = this.state
return h(
Dropdown,
{
style: {
marginLeft: '-215px',
minWidth: '180px',
},
isOpen: optionsMenuActive,
onClickOutside: (event) => {
const { classList } = event.target
const isNotToggleElement = !classList.contains(this.optionsMenuToggleClassName)
if (optionsMenuActive && isNotToggleElement) {
this.setState({ optionsMenuActive: false })
}
},
},
[
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => {
const { selected, network } = this.props
const url = genAccountLink(selected, network)
global.platform.openWindow({ url })
},
},
this.context.t('etherscanView'),
),
2017-08-04 00:32:44 +02:00
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => {
const { selected, identities } = this.props
var identity = identities[selected]
actions.showQrView(selected, identity ? identity.name : '')
},
},
this.context.t('showQRCode'),
2017-08-04 00:32:44 +02:00
),
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => {
const { selected } = this.props
copyToClipboard(checksumAddress(selected))
},
},
this.context.t('copyAddress'),
),
h(
DropdownMenuItem,
{
closeMenu: () => {},
onClick: () => {
actions.requestAccountExport()
},
},
this.context.t('exportPrivateKey'),
),
]
)
}
2017-07-18 14:25:16 +02:00
render () {
const { style, enableAccountsSelector, enableAccountOptions } = this.props
const { optionsMenuActive, accountSelectorActive } = this.state
2017-07-18 14:25:16 +02:00
return h(
'span',
{
style: style,
},
[
enableAccountsSelector && h(
// 'i.fa.fa-angle-down',
'div.cursor-pointer.color-orange.accounts-selector',
2017-07-18 14:25:16 +02:00
{
style: {
// fontSize: '1.8em',
background: 'url(images/switch_acc.svg) white center center no-repeat',
height: '25px',
width: '25px',
transform: 'scale(0.75)',
marginRight: '3px',
},
2017-07-18 14:25:16 +02:00
onClick: (event) => {
event.stopPropagation()
this.setState({
accountSelectorActive: !accountSelectorActive,
optionsMenuActive: false,
2017-07-18 14:25:16 +02:00
})
},
},
this.renderAccountSelector(),
2017-07-18 14:25:16 +02:00
),
enableAccountOptions && h(
2017-07-18 14:25:16 +02:00
'i.fa.fa-ellipsis-h',
{
style: {
marginRight: '0.5em',
fontSize: '1.8em',
},
2017-07-18 14:25:16 +02:00
onClick: (event) => {
event.stopPropagation()
this.setState({
accountSelectorActive: false,
optionsMenuActive: !optionsMenuActive,
2017-07-18 14:25:16 +02:00
})
},
},
this.renderAccountOptions()
2017-07-18 14:25:16 +02:00
),
]
)
}
}
AccountDropdowns.defaultProps = {
enableAccountsSelector: false,
enableAccountOptions: false,
}
2017-07-18 14:25:16 +02:00
AccountDropdowns.propTypes = {
identities: PropTypes.objectOf(PropTypes.object),
selected: PropTypes.string,
2017-09-07 20:51:04 +02:00
keyrings: PropTypes.array,
2017-10-02 20:08:34 +02:00
actions: PropTypes.objectOf(PropTypes.func),
network: PropTypes.string,
style: PropTypes.object,
enableAccountOptions: PropTypes.bool,
enableAccountsSelector: PropTypes.bool,
2018-03-22 03:18:10 +01:00
t: PropTypes.func,
2017-07-18 14:25:16 +02:00
}
2017-07-18 14:26:31 +02:00
const mapDispatchToProps = (dispatch) => {
2017-07-18 14:25:16 +02:00
return {
actions: {
showConfigPage: () => dispatch(actions.showConfigPage()),
requestAccountExport: () => dispatch(actions.requestExportAccount()),
showAccountDetail: (address) => dispatch(actions.showAccountDetail(address)),
addNewAccount: () => dispatch(actions.addNewAccount()),
showImportPage: () => dispatch(actions.showImportPage()),
2017-08-04 00:32:44 +02:00
showQrView: (selected, identity) => dispatch(actions.showQrView(selected, identity)),
2017-07-18 14:25:16 +02:00
},
}
}
AccountDropdowns.contextTypes = {
t: PropTypes.func,
}
2017-07-18 14:25:16 +02:00
module.exports = {
AccountDropdowns: connect(null, mapDispatchToProps)(AccountDropdowns),
}