1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-23 09:52:26 +01:00

Move account export to subview

Account detail view now has an animated transitioning `subview` section that allows us to show extra details within it.

Clicking `export` now slide replaces the transaction list with the export UI.

Added cancel/done/submit buttons to the Export UI. Done submits like Enter did, the other two transition back to the transaction list.

For some reason when first unlocking, the selected account is being instantly replaced with the accounts list, so I need to fix that before merging this into master.
This commit is contained in:
Dan Finlay 2016-05-04 18:08:31 -07:00
parent 95582f8bde
commit f4d58ebc70
4 changed files with 145 additions and 67 deletions

View File

@ -1,11 +1,15 @@
const inherits = require('util').inherits const inherits = require('util').inherits
const extend = require('xtend')
const Component = require('react').Component const Component = require('react').Component
const h = require('react-hyperscript') const h = require('react-hyperscript')
const connect = require('react-redux').connect const connect = require('react-redux').connect
const copyToClipboard = require('copy-to-clipboard') const copyToClipboard = require('copy-to-clipboard')
const actions = require('./actions') const actions = require('./actions')
const ReactCSSTransitionGroup = require('react-addons-css-transition-group')
const AccountPanel = require('./components/account-panel') const AccountPanel = require('./components/account-panel')
const transactionList = require('./components/transaction-list') const transactionList = require('./components/transaction-list')
const ExportAccountView = require('./components/account-export')
module.exports = connect(mapStateToProps)(AccountDetailScreen) module.exports = connect(mapStateToProps)(AccountDetailScreen)
@ -78,12 +82,13 @@ AccountDetailScreen.prototype.render = function() {
}, 'EXPORT'), }, 'EXPORT'),
]), ]),
transactionList(transactions h(ReactCSSTransitionGroup, {
.filter(tx => tx.txParams.from === state.address) transitionName: "main",
.filter(tx => tx.txParams.metamaskNetworkId === state.networkVersion) transitionEnterTimeout: 300,
.sort((a, b) => b.time - a.time), state.networkVersion), transitionLeaveTimeout: 300,
this.exportedAccount(accountDetail), }, [
this.subview(),
]),
// transaction table // transaction table
/* /*
h('section.flex-column', [ h('section.flex-column', [
@ -94,72 +99,41 @@ AccountDetailScreen.prototype.render = function() {
) )
} }
AccountDetailScreen.prototype.subview = function() {
var subview
try {
subview = this.props.accountDetail.subview
} catch (e) {
subview = null
}
switch (subview) {
case 'transactions':
return this.transactionList()
case 'export':
var state = extend({key: 'export'}, this.props)
return h(ExportAccountView, state)
default:
return this.transactionList()
}
}
AccountDetailScreen.prototype.transactionList = function() {
var state = this.props
var transactions = state.transactions
return transactionList(transactions
.filter(tx => tx.txParams.from === state.address)
.filter(tx => tx.txParams.metamaskNetworkId === state.networkVersion)
.sort((a, b) => b.time - a.time), state.networkVersion)
}
AccountDetailScreen.prototype.navigateToAccounts = function(event){ AccountDetailScreen.prototype.navigateToAccounts = function(event){
event.stopPropagation() event.stopPropagation()
this.props.dispatch(actions.showAccountsPage()) this.props.dispatch(actions.showAccountsPage())
} }
AccountDetailScreen.prototype.exportAccount = function(address) {
this.props.dispatch(actions.exportAccount(address))
}
AccountDetailScreen.prototype.requestAccountExport = function() { AccountDetailScreen.prototype.requestAccountExport = function() {
this.props.dispatch(actions.requestExportAccount()) this.props.dispatch(actions.requestExportAccount())
} }
AccountDetailScreen.prototype.exportedAccount = function(accountDetail) {
if (!accountDetail) return
var accountExport = accountDetail.accountExport
var notExporting = accountExport === 'none'
var exportRequested = accountExport === 'requested'
var accountExported = accountExport === 'completed'
if (notExporting) return
if (exportRequested) {
var warning = `Exporting your private key is very dangerous,
and you should only do it if you know what you're doing.`
var confirmation = `If you're absolutely sure, type "I understand" below and
hit Enter.`
return h('div', {}, [
h('p.error', warning),
h('p', confirmation),
h('input#exportAccount', {
onKeyPress: this.onExportKeyPress.bind(this),
})
])
}
if (accountExported) {
return h('div.privateKey', {
}, [
h('label', 'Your private key (click to copy):'),
h('p.error.cursor-pointer', {
style: {
textOverflow: 'ellipsis',
overflow: 'hidden',
webkitUserSelect: 'text',
width: '100%',
},
onClick: function(event) {
copyToClipboard(accountDetail.privateKey)
}
}, accountDetail.privateKey),
])
}
}
AccountDetailScreen.prototype.onExportKeyPress = function(event) {
if (event.key !== 'Enter') return
event.preventDefault()
var input = document.getElementById('exportAccount')
if (input.value === 'I understand') {
this.props.dispatch(actions.exportAccount(this.props.address))
} else {
input.value = ''
input.placeholder = 'Please retype "I understand" exactly.'
}
}

View File

@ -0,0 +1,87 @@
const Component = require('react').Component
const h = require('react-hyperscript')
const inherits = require('util').inherits
const actions = require('../actions')
module.exports = ExportAccountView
inherits(ExportAccountView, Component)
function ExportAccountView() {
Component.call(this)
}
ExportAccountView.prototype.render = function() {
console.log("EXPORT VIEW")
console.dir(this.props)
var state = this.props
var accountDetail = state.accountDetail
if (!accountDetail) return h('div')
var accountExport = accountDetail.accountExport
var notExporting = accountExport === 'none'
var exportRequested = accountExport === 'requested'
var accountExported = accountExport === 'completed'
if (notExporting) return h('div')
if (exportRequested) {
var warning = `Exporting your private key is very dangerous,
and you should only do it if you know what you're doing.`
var confirmation = `If you're absolutely sure, type "I understand" below and
submit.`
return h('div', { key: 'exporting' }, [
h('p.error', warning),
h('p', confirmation),
h('input#exportAccount', {
onKeyPress: this.onExportKeyPress.bind(this),
}),
h('button', {
onClick: () => this.onExportKeyPress({ key: 'Enter', preventDefault: () => {} }),
}, 'Submit'),
h('button', {
onClick: () => this.props.dispatch(actions.backToAccountDetail(this.props.address))
}, 'Cancel'),
])
}
if (accountExported) {
return h('div.privateKey', {
}, [
h('label', 'Your private key (click to copy):'),
h('p.error.cursor-pointer', {
style: {
textOverflow: 'ellipsis',
overflow: 'hidden',
webkitUserSelect: 'text',
width: '100%',
},
onClick: function(event) {
copyToClipboard(accountDetail.privateKey)
}
}, accountDetail.privateKey),
h('button', {
onClick: () => this.props.dispatch(actions.backToAccountDetail(this.props.address))
}, 'Done'),
])
}
}
ExportAccountView.prototype.onExportKeyPress = function(event) {
if (event.key !== 'Enter') return
event.preventDefault()
var input = document.getElementById('exportAccount')
if (input.value === 'I understand') {
this.props.dispatch(actions.exportAccount(this.props.address))
} else {
input.value = ''
input.placeholder = 'Please retype "I understand" exactly.'
}
}
ExportAccountView.prototype.exportAccount = function(address) {
this.props.dispatch(actions.exportAccount(address))
}

View File

@ -4,7 +4,7 @@ const addressSummary = require('../util').addressSummary
const explorerLink = require('../../lib/explorer-link') const explorerLink = require('../../lib/explorer-link')
module.exports = function(transactions, network) { module.exports = function(transactions, network) {
return h('details', [ return h('details', { key: 'transaction-list' }, [
h('summary', [ h('summary', [
h('div.font-small', {style: {display: 'inline'}}, 'Transactions'), h('div.font-small', {style: {display: 'inline'}}, 'Transactions'),

View File

@ -23,6 +23,9 @@ function reduceApp(state, action) {
var appState = extend({ var appState = extend({
currentView: seedWords ? seedConfView : defaultView, currentView: seedWords ? seedConfView : defaultView,
accountDetail: {
subview: 'transactions',
},
currentDomain: 'example.com', currentDomain: 'example.com',
transForward: true, // Used to render transition direction transForward: true, // Used to render transition direction
isLoading: false, // Used to display loading indicator isLoading: false, // Used to display loading indicator
@ -131,6 +134,7 @@ function reduceApp(state, action) {
return extend(appState, { return extend(appState, {
currentView: {}, currentView: {},
accountDetail: { accountDetail: {
subview: 'transactions',
accountExport: 'none', accountExport: 'none',
privateKey: '', privateKey: '',
}, },
@ -144,6 +148,7 @@ function reduceApp(state, action) {
context: action.value || account, context: action.value || account,
}, },
accountDetail: { accountDetail: {
subview: 'transactions',
accountExport: 'none', accountExport: 'none',
privateKey: '', privateKey: '',
}, },
@ -157,6 +162,7 @@ function reduceApp(state, action) {
context: action.value, context: action.value,
}, },
accountDetail: { accountDetail: {
subview: 'transactions',
accountExport: 'none', accountExport: 'none',
privateKey: '', privateKey: '',
}, },
@ -218,6 +224,9 @@ function reduceApp(state, action) {
name: 'accountDetail', name: 'accountDetail',
context: state.metamask.selectedAddress, context: state.metamask.selectedAddress,
}, },
accountDetail: {
subview: 'transactions',
},
}) })
} }
@ -285,7 +294,13 @@ function reduceApp(state, action) {
case actions.REQUEST_ACCOUNT_EXPORT: case actions.REQUEST_ACCOUNT_EXPORT:
return extend(appState, { return extend(appState, {
transForward: true,
currentView: {
name: 'accountDetail',
context: appState.currentView.context,
},
accountDetail: { accountDetail: {
subview: 'export',
accountExport: 'requested', accountExport: 'requested',
}, },
}) })
@ -293,6 +308,7 @@ function reduceApp(state, action) {
case actions.EXPORT_ACCOUNT: case actions.EXPORT_ACCOUNT:
return extend(appState, { return extend(appState, {
accountDetail: { accountDetail: {
subview: 'export',
accountExport: 'completed', accountExport: 'completed',
}, },
}) })
@ -300,6 +316,7 @@ function reduceApp(state, action) {
case actions.SHOW_PRIVATE_KEY: case actions.SHOW_PRIVATE_KEY:
return extend(appState, { return extend(appState, {
accountDetail: { accountDetail: {
subview: 'export',
accountExport: 'completed', accountExport: 'completed',
privateKey: action.value, privateKey: action.value,
}, },