Browse Source

Merge pull request #465 from gre/fix-send-flow

Fix send flow
master
Meriadec Pillet 7 years ago
committed by GitHub
parent
commit
8b5e088a50
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 97
      src/bridge/EthereumJSBridge.js
  2. 26
      src/bridge/LibcoreBridge.js
  3. 141
      src/bridge/RippleJSBridge.js
  4. 6
      src/bridge/UnsupportedBridge.js
  5. 31
      src/bridge/makeMockBridge.js
  6. 3
      src/bridge/types.js
  7. 150
      src/commands/libcoreSignAndBroadcast.js
  8. 42
      src/components/DeviceSignTransaction.js
  9. 4
      src/components/SelectAccount/index.js
  10. 38
      src/components/modals/Send/03-step-verification.js
  11. 13
      src/components/modals/Send/04-step-confirmation.js
  12. 53
      src/components/modals/Send/ConfirmationFooter.js
  13. 98
      src/components/modals/Send/SendModalBody.js
  14. 16
      src/components/modals/Send/index.js
  15. 2
      static/i18n/en/send.yml

97
src/bridge/EthereumJSBridge.js

@ -1,4 +1,5 @@
// @flow // @flow
import { Observable } from 'rxjs'
import React from 'react' import React from 'react'
import FeesField from 'components/FeesField/EthereumKind' import FeesField from 'components/FeesField/EthereumKind'
import AdvancedOptions from 'components/AdvancedOptions/EthereumKind' import AdvancedOptions from 'components/AdvancedOptions/EthereumKind'
@ -93,6 +94,49 @@ function mergeOps(existing: Operation[], newFetched: Operation[]) {
return uniqBy(all.sort((a, b) => b.date - a.date), 'id') return uniqBy(all.sort((a, b) => b.date - a.date), 'id')
} }
const signAndBroadcast = async ({
a,
t,
deviceId,
isCancelled,
onSigned,
onOperationBroadcasted,
}) => {
const api = apiForCurrency(a.currency)
const nonce = await api.getAccountNonce(a.freshAddress)
const transaction = await signTransactionCommand
.send({
currencyId: a.currency.id,
devicePath: deviceId,
path: a.freshAddressPath,
transaction: { ...t, nonce },
})
.toPromise()
if (!isCancelled()) {
onSigned()
const hash = await api.broadcastTransaction(transaction)
onOperationBroadcasted({
id: `${a.id}-${hash}-OUT`,
hash,
type: 'OUT',
value: t.amount,
fee: t.gasPrice * t.gasLimit,
blockHeight: null,
blockHash: null,
accountId: a.id,
senders: [a.freshAddress],
recipients: [t.recipient],
transactionSequenceNumber: nonce,
date: new Date(),
})
}
}
const SAFE_REORG_THRESHOLD = 80 const SAFE_REORG_THRESHOLD = 80
const fetchCurrentBlock = (perCurrencyId => currency => { const fetchCurrentBlock = (perCurrencyId => currency => {
@ -324,37 +368,28 @@ const EthereumBridge: WalletBridge<Transaction> = {
getTotalSpent: (a, t) => Promise.resolve(t.amount + t.gasPrice * t.gasLimit), getTotalSpent: (a, t) => Promise.resolve(t.amount + t.gasPrice * t.gasLimit),
getMaxAmount: (a, t) => Promise.resolve(a.balance - t.gasPrice * t.gasLimit), getMaxAmount: (a, t) => Promise.resolve(a.balance - t.gasPrice * t.gasLimit),
signAndBroadcast: async (a, t, deviceId) => { signAndBroadcast: (a, t, deviceId) =>
const api = apiForCurrency(a.currency) Observable.create(o => {
let cancelled = false
const nonce = await api.getAccountNonce(a.freshAddress) const isCancelled = () => cancelled
const onSigned = () => {
const transaction = await signTransactionCommand o.next({ type: 'signed' })
.send({ }
currencyId: a.currency.id, const onOperationBroadcasted = operation => {
devicePath: deviceId, o.next({ type: 'broadcasted', operation })
path: a.freshAddressPath, }
transaction: { ...t, nonce }, signAndBroadcast({ a, t, deviceId, isCancelled, onSigned, onOperationBroadcasted }).then(
}) () => {
.toPromise() o.complete()
},
const hash = await api.broadcastTransaction(transaction) e => {
o.error(e)
return { },
id: `${a.id}-${hash}-OUT`, )
hash, return () => {
type: 'OUT', cancelled = true
value: t.amount, }
fee: t.gasPrice * t.gasLimit, }),
blockHeight: null,
blockHash: null,
accountId: a.id,
senders: [a.freshAddress],
recipients: [t.recipient],
transactionSequenceNumber: nonce,
date: new Date(),
}
},
addPendingOperation: (account, operation) => ({ addPendingOperation: (account, operation) => ({
...account, ...account,

26
src/bridge/LibcoreBridge.js

@ -17,6 +17,9 @@ type Transaction = {
recipient: string, recipient: string,
} }
const decodeOperation = (encodedAccount, rawOp) =>
decodeAccount({ ...encodedAccount, operations: [rawOp] }).operations[0]
const EditFees = ({ account, onChange, value }: EditProps<Transaction>) => ( const EditFees = ({ account, onChange, value }: EditProps<Transaction>) => (
<FeesBitcoinKind <FeesBitcoinKind
onChange={feePerByte => { onChange={feePerByte => {
@ -135,20 +138,27 @@ const LibcoreBridge: WalletBridge<Transaction> = {
getMaxAmount: (a, _t) => Promise.resolve(a.balance), // FIXME getMaxAmount: (a, _t) => Promise.resolve(a.balance), // FIXME
signAndBroadcast: async (account, transaction, deviceId) => { signAndBroadcast: (account, transaction, deviceId) => {
const encodedAccount = encodeAccount(account) const encodedAccount = encodeAccount(account)
const rawOp = await libcoreSignAndBroadcast return libcoreSignAndBroadcast
.send({ .send({
account: encodedAccount, account: encodedAccount,
transaction, transaction,
deviceId, deviceId,
}) })
.toPromise() .pipe(
map(e => {
// quick HACK switch (e.type) {
const [op] = decodeAccount({ ...encodedAccount, operations: [rawOp] }).operations case 'broadcasted':
return {
return op type: 'broadcasted',
operation: decodeOperation(encodedAccount, e.operation),
}
default:
return e
}
}),
)
}, },
addPendingOperation: (account, operation) => ({ addPendingOperation: (account, operation) => ({

141
src/bridge/RippleJSBridge.js

@ -1,4 +1,5 @@
// @flow // @flow
import { Observable } from 'rxjs'
import React from 'react' import React from 'react'
import bs58check from 'ripple-bs58check' import bs58check from 'ripple-bs58check'
import { computeBinaryTransactionHash } from 'ripple-hashes' import { computeBinaryTransactionHash } from 'ripple-hashes'
@ -43,6 +44,70 @@ const EditAdvancedOptions = ({ onChange, value }: EditProps<Transaction>) => (
/> />
) )
async function signAndBroadcast({ a, t, deviceId, isCancelled, onSigned, onOperationBroadcasted }) {
const api = apiForCurrency(a.currency)
try {
await api.connect()
const amount = formatAPICurrencyXRP(t.amount)
const payment = {
source: {
address: a.freshAddress,
amount,
},
destination: {
address: t.recipient,
minAmount: amount,
tag: t.tag,
},
}
const instruction = {
fee: formatAPICurrencyXRP(t.fee).value,
}
const prepared = await api.preparePayment(a.freshAddress, payment, instruction)
const transaction = await signTransaction
.send({
currencyId: a.currency.id,
devicePath: deviceId,
path: a.freshAddressPath,
transaction: JSON.parse(prepared.txJSON),
})
.toPromise()
if (!isCancelled()) {
onSigned()
const submittedPayment = await api.submit(transaction)
if (submittedPayment.resultCode !== 'tesSUCCESS') {
throw new Error(submittedPayment.resultMessage)
}
const hash = computeBinaryTransactionHash(transaction)
onOperationBroadcasted({
id: `${a.id}-${hash}-OUT`,
hash,
accountId: a.id,
type: 'OUT',
value: t.amount,
fee: t.fee,
blockHash: null,
blockHeight: null,
senders: [a.freshAddress],
recipients: [t.recipient],
date: new Date(),
// we probably can't get it so it's a predictive value
transactionSequenceNumber:
(a.operations.length > 0 ? a.operations[0].transactionSequenceNumber : 0) +
a.pendingOperations.length,
})
}
} finally {
api.disconnect()
}
}
function isRecipientValid(currency, recipient) { function isRecipientValid(currency, recipient) {
try { try {
bs58check.decode(recipient) bs58check.decode(recipient)
@ -394,66 +459,28 @@ const RippleJSBridge: WalletBridge<Transaction> = {
getMaxAmount: (a, t) => Promise.resolve(a.balance - t.fee), getMaxAmount: (a, t) => Promise.resolve(a.balance - t.fee),
signAndBroadcast: async (a, t, deviceId) => { signAndBroadcast: (a, t, deviceId) =>
const api = apiForCurrency(a.currency) Observable.create(o => {
try { let cancelled = false
await api.connect() const isCancelled = () => cancelled
const amount = formatAPICurrencyXRP(t.amount) const onSigned = () => {
const payment = { o.next({ type: 'signed' })
source: {
address: a.freshAddress,
amount,
},
destination: {
address: t.recipient,
minAmount: amount,
tag: t.tag,
},
} }
const instruction = { const onOperationBroadcasted = operation => {
fee: formatAPICurrencyXRP(t.fee).value, o.next({ type: 'broadcasted', operation })
}
const prepared = await api.preparePayment(a.freshAddress, payment, instruction)
const transaction = await signTransaction
.send({
currencyId: a.currency.id,
devicePath: deviceId,
path: a.freshAddressPath,
transaction: JSON.parse(prepared.txJSON),
})
.toPromise()
const submittedPayment = await api.submit(transaction)
if (submittedPayment.resultCode !== 'tesSUCCESS') {
throw new Error(submittedPayment.resultMessage)
} }
signAndBroadcast({ a, t, deviceId, isCancelled, onSigned, onOperationBroadcasted }).then(
const hash = computeBinaryTransactionHash(transaction) () => {
o.complete()
return { },
id: `${a.id}-${hash}-OUT`, e => {
hash, o.error(e)
accountId: a.id, },
type: 'OUT', )
value: t.amount, return () => {
fee: t.fee, cancelled = true
blockHash: null,
blockHeight: null,
senders: [a.freshAddress],
recipients: [t.recipient],
date: new Date(),
// we probably can't get it so it's a predictive value
transactionSequenceNumber:
(a.operations.length > 0 ? a.operations[0].transactionSequenceNumber : 0) +
a.pendingOperations.length,
} }
} finally { }),
api.disconnect()
}
},
addPendingOperation: (account, operation) => ({ addPendingOperation: (account, operation) => ({
...account, ...account,

6
src/bridge/UnsupportedBridge.js

@ -1,4 +1,5 @@
// @flow // @flow
import { Observable } from 'rxjs'
import type { WalletBridge } from './types' import type { WalletBridge } from './types'
const genericError = new Error('UnsupportedBridge') const genericError = new Error('UnsupportedBridge')
@ -36,7 +37,10 @@ const UnsupportedBridge: WalletBridge<*> = {
getMaxAmount: () => Promise.resolve(0), getMaxAmount: () => Promise.resolve(0),
signAndBroadcast: () => Promise.reject(genericError), signAndBroadcast: () =>
Observable.create(o => {
o.error(genericError)
}),
} }
export default UnsupportedBridge export default UnsupportedBridge

31
src/bridge/makeMockBridge.js

@ -1,4 +1,5 @@
// @flow // @flow
import { Observable } from 'rxjs'
import { import {
genAccount, genAccount,
genAddingOperationsInAccount, genAddingOperationsInAccount,
@ -149,20 +150,22 @@ function makeMockBridge(opts?: Opts): WalletBridge<*> {
getMaxAmount, getMaxAmount,
signAndBroadcast: async (account, t) => { signAndBroadcast: (account, t) =>
const rng = new Prando() Observable.create(o => {
const op = genOperation(account, account.operations, account.currency, rng) const rng = new Prando()
op.type = 'OUT' const op = genOperation(account, account.operations, account.currency, rng)
op.value = t.amount op.type = 'OUT'
op.blockHash = null op.value = t.amount
op.blockHeight = null op.blockHash = null
op.senders = [account.freshAddress] op.blockHeight = null
op.recipients = [t.recipient] op.senders = [account.freshAddress]
op.blockHeight = account.blockHeight op.recipients = [t.recipient]
op.date = new Date() op.blockHeight = account.blockHeight
broadcasted[account.id] = (broadcasted[account.id] || []).concat(op) op.date = new Date()
return { ...op } broadcasted[account.id] = (broadcasted[account.id] || []).concat(op)
}, o.next({ type: 'signed' })
o.next({ type: 'broadcasted', operation: { ...op } })
}),
} }
} }

3
src/bridge/types.js

@ -1,5 +1,6 @@
// @flow // @flow
import type { Observable } from 'rxjs'
import type { Account, Operation, Currency } from '@ledgerhq/live-common/lib/types' import type { Account, Operation, Currency } from '@ledgerhq/live-common/lib/types'
// a WalletBridge is implemented on renderer side. // a WalletBridge is implemented on renderer side.
@ -104,7 +105,7 @@ export interface WalletBridge<Transaction> {
account: Account, account: Account,
transaction: Transaction, transaction: Transaction,
deviceId: DeviceId, deviceId: DeviceId,
): Promise<Operation>; ): Observable<{ type: 'signed' } | { type: 'broadcasted', operation: Operation }>;
// Implement an optimistic response for signAndBroadcast. // Implement an optimistic response for signAndBroadcast.
// you likely should add the operation in account.pendingOperations but maybe you want to clean it (because maybe some are replaced / cancelled by this one?) // you likely should add the operation in account.pendingOperations but maybe you want to clean it (because maybe some are replaced / cancelled by this one?)

150
src/commands/libcoreSignAndBroadcast.js

@ -2,9 +2,8 @@
import type { AccountRaw, OperationRaw } from '@ledgerhq/live-common/lib/types' import type { AccountRaw, OperationRaw } from '@ledgerhq/live-common/lib/types'
import Btc from '@ledgerhq/hw-app-btc' import Btc from '@ledgerhq/hw-app-btc'
import { fromPromise } from 'rxjs/observable/fromPromise' import { Observable } from 'rxjs'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies' import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies'
import type Transport from '@ledgerhq/hw-transport'
import withLibcore from 'helpers/withLibcore' import withLibcore from 'helpers/withLibcore'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
@ -23,24 +22,37 @@ type Input = {
deviceId: string, deviceId: string,
} }
type Result = $Exact<OperationRaw> type Result = { type: 'signed' } | { type: 'broadcasted', operation: OperationRaw }
const cmd: Command<Input, Result> = createCommand( const cmd: Command<Input, Result> = createCommand(
'libcoreSignAndBroadcast', 'libcoreSignAndBroadcast',
({ account, transaction, deviceId }) => ({ account, transaction, deviceId }) =>
fromPromise( Observable.create(o => {
withDevice(deviceId)(transport => let unsubscribed = false
withLibcore(core => const isCancelled = () => unsubscribed
doSignAndBroadcast({ withLibcore(core =>
account, doSignAndBroadcast({
transaction, account,
deviceId, transaction,
core, deviceId,
transport, core,
}), isCancelled,
), onSigned: () => {
), o.next({ type: 'signed' })
), },
onOperationBroadcasted: operation => {
o.next({
type: 'broadcasted',
operation,
})
},
}),
).then(() => o.complete(), e => o.error(e))
return () => {
unsubscribed = true
}
}),
) )
export async function doSignAndBroadcast({ export async function doSignAndBroadcast({
@ -48,60 +60,74 @@ export async function doSignAndBroadcast({
transaction, transaction,
deviceId, deviceId,
core, core,
transport, isCancelled,
onSigned,
onOperationBroadcasted,
}: { }: {
account: AccountRaw, account: AccountRaw,
transaction: BitcoinLikeTransaction, transaction: BitcoinLikeTransaction,
deviceId: string, deviceId: string,
core: *, core: *,
transport: Transport<*>, isCancelled: () => boolean,
}) { onSigned: () => void,
const hwApp = new Btc(transport) onOperationBroadcasted: (optimisticOp: $Exact<OperationRaw>) => void,
}): Promise<void> {
const WALLET_IDENTIFIER = await getWalletIdentifier({ let njsAccount
hwApp,
isSegwit: !!account.isSegwit, const signedTransaction: ?string = await withDevice(deviceId)(async transport => {
currencyId: account.currencyId, const hwApp = new Btc(transport)
devicePath: deviceId,
const WALLET_IDENTIFIER = await getWalletIdentifier({
hwApp,
isSegwit: !!account.isSegwit,
currencyId: account.currencyId,
devicePath: deviceId,
})
const njsWallet = await core.getWallet(WALLET_IDENTIFIER)
if (isCancelled()) return null
njsAccount = await njsWallet.getAccount(account.index)
if (isCancelled()) return null
const bitcoinLikeAccount = njsAccount.asBitcoinLikeAccount()
const njsWalletCurrency = njsWallet.getCurrency()
const amount = core.createAmount(njsWalletCurrency, transaction.amount)
const fees = core.createAmount(njsWalletCurrency, transaction.feePerByte)
const transactionBuilder = bitcoinLikeAccount.buildTransaction()
// TODO: check if is valid address. if not, it will fail silently on invalid
transactionBuilder.sendToAddress(amount, transaction.recipient)
// TODO: don't use hardcoded value for sequence (and first also maybe)
transactionBuilder.pickInputs(0, 0xffffff)
transactionBuilder.setFeesPerByte(fees)
const builded = await transactionBuilder.build()
if (isCancelled()) return null
const sigHashType = core.helpers.bytesToHex(
njsWalletCurrency.bitcoinLikeNetworkParameters.SigHash,
)
const hasTimestamp = njsWalletCurrency.bitcoinLikeNetworkParameters.UsesTimestampedTransaction
// TODO: const timestampDelay = njsWalletCurrency.bitcoinLikeNetworkParameters.TimestampDelay
const currency = getCryptoCurrencyById(account.currencyId)
return core.signTransaction({
hwApp,
transaction: builded,
sigHashType: parseInt(sigHashType, 16).toString(),
supportsSegwit: !!currency.supportsSegwit,
isSegwit: account.isSegwit,
hasTimestamp,
})
}) })
const njsWallet = await core.getWallet(WALLET_IDENTIFIER) if (!signedTransaction || isCancelled() || !njsAccount) return
const njsAccount = await njsWallet.getAccount(account.index) onSigned()
const bitcoinLikeAccount = njsAccount.asBitcoinLikeAccount()
const njsWalletCurrency = njsWallet.getCurrency()
const amount = core.createAmount(njsWalletCurrency, transaction.amount)
const fees = core.createAmount(njsWalletCurrency, transaction.feePerByte)
const transactionBuilder = bitcoinLikeAccount.buildTransaction()
// TODO: check if is valid address. if not, it will fail silently on invalid
transactionBuilder.sendToAddress(amount, transaction.recipient)
// TODO: don't use hardcoded value for sequence (and first also maybe)
transactionBuilder.pickInputs(0, 0xffffff)
transactionBuilder.setFeesPerByte(fees)
const builded = await transactionBuilder.build()
const sigHashType = core.helpers.bytesToHex(
njsWalletCurrency.bitcoinLikeNetworkParameters.SigHash,
)
const hasTimestamp = njsWalletCurrency.bitcoinLikeNetworkParameters.UsesTimestampedTransaction
// TODO: const timestampDelay = njsWalletCurrency.bitcoinLikeNetworkParameters.TimestampDelay
const currency = getCryptoCurrencyById(account.currencyId)
const signedTransaction = await core.signTransaction({
hwApp,
transaction: builded,
sigHashType: parseInt(sigHashType, 16).toString(),
supportsSegwit: !!currency.supportsSegwit,
isSegwit: account.isSegwit,
hasTimestamp,
})
const txHash = await njsAccount.asBitcoinLikeAccount().broadcastRawTransaction(signedTransaction) const txHash = await njsAccount.asBitcoinLikeAccount().broadcastRawTransaction(signedTransaction)
// NB we don't check isCancelled() because the broadcast is not cancellable now!
// optimistic operation onOperationBroadcasted({
return {
id: txHash, id: txHash,
hash: txHash, hash: txHash,
type: 'OUT', type: 'OUT',
@ -113,7 +139,7 @@ export async function doSignAndBroadcast({
recipients: [transaction.recipient], recipients: [transaction.recipient],
accountId: account.id, accountId: account.id,
date: new Date().toISOString(), date: new Date().toISOString(),
} })
} }
export default cmd export default cmd

42
src/components/DeviceSignTransaction.js

@ -1,42 +0,0 @@
// @flow
import { PureComponent } from 'react'
import type { Account, Operation } from '@ledgerhq/live-common/lib/types'
import type { Device } from 'types/common'
import type { WalletBridge } from 'bridge/types'
type Props = {
children: *,
onOperationBroadcasted: (op: Operation) => void,
onError: Error => void,
device: Device,
account: Account,
bridge: WalletBridge<*>,
transaction: *,
}
class DeviceSignTransaction extends PureComponent<Props> {
componentDidMount() {
this.sign()
}
componentWillUnmount() {
this.unmount = true
}
unmount = false
sign = async () => {
const { device, account, transaction, bridge, onOperationBroadcasted, onError } = this.props
try {
const optimisticOperation = await bridge.signAndBroadcast(account, transaction, device.path)
onOperationBroadcasted(optimisticOperation)
} catch (error) {
onError(error)
}
}
render() {
return this.props.children
}
}
export default DeviceSignTransaction

4
src/components/SelectAccount/index.js

@ -57,9 +57,7 @@ type Props = {
} }
const RawSelectAccount = ({ accounts, onChange, value, t, ...props }: Props) => { const RawSelectAccount = ({ accounts, onChange, value, t, ...props }: Props) => {
const options = accounts const options = accounts.map(a => ({ ...a, value: a.id, label: a.name }))
.sort((a, b) => (a.name < b.name ? -1 : 1))
.map(a => ({ ...a, value: a.id, label: a.name }))
const selectedOption = value ? options.find(o => o.value === value.id) : null const selectedOption = value ? options.find(o => o.value === value.id) : null
return ( return (
<Select <Select

38
src/components/modals/Send/03-step-verification.js

@ -6,12 +6,9 @@ import styled from 'styled-components'
import Box from 'components/base/Box' import Box from 'components/base/Box'
import WarnBox from 'components/WarnBox' import WarnBox from 'components/WarnBox'
import { multiline } from 'styles/helpers' import { multiline } from 'styles/helpers'
import DeviceSignTransaction from 'components/DeviceSignTransaction'
import DeviceConfirm from 'components/DeviceConfirm' import DeviceConfirm from 'components/DeviceConfirm'
import type { WalletBridge } from 'bridge/types' import type { T } from 'types/common'
import type { Account, Operation } from '@ledgerhq/live-common/lib/types'
import type { Device, T } from 'types/common'
const Container = styled(Box).attrs({ const Container = styled(Box).attrs({
alignItems: 'center', alignItems: 'center',
@ -30,43 +27,14 @@ const Info = styled(Box).attrs({
` `
type Props = { type Props = {
account: ?Account,
device: ?Device,
bridge: ?WalletBridge<*>,
transaction: *,
onOperationBroadcasted: (op: Operation) => void,
onError: (e: Error) => void,
hasError: boolean, hasError: boolean,
t: T, t: T,
} }
export default ({ export default ({ t, hasError }: Props) => (
account,
device,
bridge,
transaction,
onOperationBroadcasted,
t,
onError,
hasError,
}: Props) => (
<Container> <Container>
<WarnBox>{multiline(t('send:steps.verification.warning'))}</WarnBox> <WarnBox>{multiline(t('send:steps.verification.warning'))}</WarnBox>
<Info>{t('send:steps.verification.body')}</Info> <Info>{t('send:steps.verification.body')}</Info>
{account && <DeviceConfirm error={hasError} />
bridge &&
transaction &&
device && (
<DeviceSignTransaction
account={account}
device={device}
transaction={transaction}
bridge={bridge}
onOperationBroadcasted={onOperationBroadcasted}
onError={onError}
>
<DeviceConfirm error={hasError} />
</DeviceSignTransaction>
)}
</Container> </Container>
) )

13
src/components/modals/Send/04-step-confirmation.js

@ -3,6 +3,7 @@ import React from 'react'
import styled from 'styled-components' import styled from 'styled-components'
import type { Operation } from '@ledgerhq/live-common/lib/types' import type { Operation } from '@ledgerhq/live-common/lib/types'
import Spinner from 'components/base/Spinner'
import IconCheckCircle from 'icons/CheckCircle' import IconCheckCircle from 'icons/CheckCircle'
import IconExclamationCircleThin from 'icons/ExclamationCircleThin' import IconExclamationCircleThin from 'icons/ExclamationCircleThin'
import Box from 'components/base/Box' import Box from 'components/base/Box'
@ -45,11 +46,17 @@ type Props = {
function StepConfirmation(props: Props) { function StepConfirmation(props: Props) {
const { t, optimisticOperation, error } = props const { t, optimisticOperation, error } = props
const Icon = optimisticOperation ? IconCheckCircle : IconExclamationCircleThin const Icon = optimisticOperation ? IconCheckCircle : error ? IconExclamationCircleThin : Spinner
const iconColor = optimisticOperation ? colors.positiveGreen : colors.alertRed const iconColor = optimisticOperation
? colors.positiveGreen
: error
? colors.alertRed
: colors.grey
const tPrefix = optimisticOperation const tPrefix = optimisticOperation
? 'send:steps.confirmation.success' ? 'send:steps.confirmation.success'
: 'send:steps.confirmation.error' : error
? 'send:steps.confirmation.error'
: 'send:steps.confirmation.pending'
return ( return (
<Container> <Container>

53
src/components/modals/Send/ConfirmationFooter.js

@ -9,37 +9,42 @@ import type { T } from 'types/common'
export default ({ export default ({
t, t,
error,
account, account,
optimisticOperation, optimisticOperation,
onClose, onClose,
onGoToFirstStep, onGoToFirstStep,
}: { }: {
t: T, t: T,
error: ?Error,
account: ?Account, account: ?Account,
optimisticOperation: ?Operation, optimisticOperation: ?Operation,
onClose: () => void, onClose: () => void,
onGoToFirstStep: () => void, onGoToFirstStep: () => void,
}) => ( }) => {
<ModalFooter horizontal alignItems="center" justifyContent="flex-end" flow={2}> const url =
<Button onClick={onClose}>{t('common:close')}</Button> optimisticOperation && account && getAccountOperationExplorer(account, optimisticOperation)
{optimisticOperation ? ( return (
// TODO: actually go to operations details <ModalFooter horizontal alignItems="center" justifyContent="flex-end" flow={2}>
<Button <Button onClick={onClose}>{t('common:close')}</Button>
onClick={() => { {optimisticOperation ? (
const url = account && getAccountOperationExplorer(account, optimisticOperation) // TODO: actually go to operations details
if (url) { url ? (
shell.openExternal(url) <Button
} onClick={() => {
onClose() shell.openExternal(url)
}} onClose()
primary }}
> primary
{t('send:steps.confirmation.success.cta')} >
</Button> {t('send:steps.confirmation.success.cta')}
) : ( </Button>
<Button onClick={onGoToFirstStep} primary> ) : null
{t('send:steps.confirmation.error.cta')} ) : error ? (
</Button> <Button onClick={onGoToFirstStep} primary>
)} {t('send:steps.confirmation.error.cta')}
</ModalFooter> </Button>
) ) : null}
</ModalFooter>
)
}

98
src/components/modals/Send/SendModalBody.js

@ -1,5 +1,6 @@
// @flow // @flow
import invariant from 'invariant'
import React, { PureComponent } from 'react' import React, { PureComponent } from 'react'
import { translate } from 'react-i18next' import { translate } from 'react-i18next'
import { connect } from 'react-redux' import { connect } from 'react-redux'
@ -27,6 +28,8 @@ import StepAmount from './01-step-amount'
import StepVerification from './03-step-verification' import StepVerification from './03-step-verification'
import StepConfirmation from './04-step-confirmation' import StepConfirmation from './04-step-confirmation'
const noop = () => {}
type Props = { type Props = {
initialAccount: ?Account, initialAccount: ?Account,
onClose: () => void, onClose: () => void,
@ -48,7 +51,9 @@ type State<T> = {
type Step = { type Step = {
label: string, label: string,
canNext?: (State<*>) => boolean, canNext: (State<*>) => boolean,
canPrev: (State<*>) => boolean,
canClose: (State<*>) => boolean,
prevStep?: number, prevStep?: number,
} }
@ -81,6 +86,8 @@ class SendModalBody extends PureComponent<Props, State<*>> {
this.steps = [ this.steps = [
{ {
label: t('send:steps.amount.title'), label: t('send:steps.amount.title'),
canClose: () => true,
canPrev: () => false,
canNext: ({ bridge, account, transaction }) => canNext: ({ bridge, account, transaction }) =>
bridge && account && transaction bridge && account && transaction
? bridge.isValidTransaction(account, transaction) ? bridge.isValidTransaction(account, transaction)
@ -88,28 +95,83 @@ class SendModalBody extends PureComponent<Props, State<*>> {
}, },
{ {
label: t('send:steps.connectDevice.title'), label: t('send:steps.connectDevice.title'),
canClose: () => true,
canNext: ({ deviceSelected, appStatus }) => canNext: ({ deviceSelected, appStatus }) =>
deviceSelected !== null && appStatus === 'success', deviceSelected !== null && appStatus === 'success',
prevStep: 0, prevStep: 0,
canPrev: () => true,
}, },
{ {
label: t('send:steps.verification.title'), label: t('send:steps.verification.title'),
canClose: ({ error }) => !!error,
canNext: () => true, canNext: () => true,
canPrev: ({ error }) => !!error,
prevStep: 1, prevStep: 1,
}, },
{ {
label: t('send:steps.confirmation.title'), label: t('send:steps.confirmation.title'),
prevStep: 0, prevStep: 0,
canClose: () => true,
canPrev: () => true,
canNext: () => false,
}, },
] ]
} }
componentWillUnmount() {
const { signTransactionSub } = this
if (signTransactionSub) {
signTransactionSub.unsubscribe()
}
}
signTransactionSub: *
signTransaction = async () => {
const { deviceSelected, account, transaction, bridge } = this.state
invariant(
deviceSelected && account && transaction && bridge,
'signTransaction invalid conditions',
)
this.signTransactionSub = bridge
.signAndBroadcast(account, transaction, deviceSelected.path)
.subscribe({
next: e => {
switch (e.type) {
case 'signed': {
this.onSigned()
break
}
case 'broadcasted': {
this.onOperationBroadcasted(e.operation)
break
}
default:
}
},
error: error => {
this.onOperationError(error)
},
})
}
onNextStep = () => onNextStep = () =>
this.setState(({ stepIndex }) => { this.setState(state => {
let { stepIndex, error } = state
if (stepIndex >= this.steps.length - 1) { if (stepIndex >= this.steps.length - 1) {
return null return null
} }
return { stepIndex: stepIndex + 1 } if (!this.steps[stepIndex].canNext(state)) {
console.warn('tried to next step without a valid state!', state, stepIndex)
return null
}
stepIndex++
if (stepIndex < 2) {
error = null
} else if (stepIndex === 2) {
this.signTransaction()
}
return { stepIndex, error }
}) })
onChangeDevice = (deviceSelected: ?Device) => { onChangeDevice = (deviceSelected: ?Device) => {
@ -133,8 +195,14 @@ class SendModalBody extends PureComponent<Props, State<*>> {
} }
} }
onSigned = () => {
this.setState({
stepIndex: 3,
})
}
onOperationBroadcasted = (optimisticOperation: Operation) => { onOperationBroadcasted = (optimisticOperation: Operation) => {
const { stepIndex, account, bridge } = this.state const { account, bridge } = this.state
if (!account || !bridge) return if (!account || !bridge) return
const { addPendingOperation } = bridge const { addPendingOperation } = bridge
if (addPendingOperation) { if (addPendingOperation) {
@ -144,7 +212,7 @@ class SendModalBody extends PureComponent<Props, State<*>> {
} }
this.setState({ this.setState({
optimisticOperation, optimisticOperation,
stepIndex: stepIndex + 1, stepIndex: 3,
error: null, error: null,
}) })
} }
@ -179,7 +247,7 @@ class SendModalBody extends PureComponent<Props, State<*>> {
steps: Step[] steps: Step[]
render() { render() {
const { t, onClose } = this.props const { t } = this.props
const { const {
stepIndex, stepIndex,
account, account,
@ -192,8 +260,13 @@ class SendModalBody extends PureComponent<Props, State<*>> {
const step = this.steps[stepIndex] const step = this.steps[stepIndex]
if (!step) return null if (!step) return null
const canNext = step.canNext && step.canNext(this.state) const canClose = step.canClose(this.state)
const canPrev = 'prevStep' in step const canNext = step.canNext(this.state)
const canPrev = step.canPrev(this.state)
let { onClose } = this.props
if (!canClose) {
onClose = noop
}
return ( return (
<ModalBody onClose={onClose}> <ModalBody onClose={onClose}>
@ -242,6 +315,7 @@ class SendModalBody extends PureComponent<Props, State<*>> {
{stepIndex === 3 ? ( {stepIndex === 3 ? (
<ConfirmationFooter <ConfirmationFooter
t={t} t={t}
error={error}
account={account} account={account}
optimisticOperation={optimisticOperation} optimisticOperation={optimisticOperation}
onClose={onClose} onClose={onClose}
@ -268,4 +342,10 @@ class SendModalBody extends PureComponent<Props, State<*>> {
} }
} }
export default compose(connect(mapStateToProps, mapDispatchToProps), translate())(SendModalBody) export default compose(
connect(
mapStateToProps,
mapDispatchToProps,
),
translate(),
)(SendModalBody)

16
src/components/modals/Send/index.js

@ -4,24 +4,14 @@ import { MODAL_SEND } from 'config/constants'
import Modal from 'components/base/Modal' import Modal from 'components/base/Modal'
import SendModalBody from './SendModalBody' import SendModalBody from './SendModalBody'
class SendModal extends PureComponent<{}, { resetId: number }> { class SendModal extends PureComponent<{}> {
state = { resetId: 0 }
handleReset = () => {
this.setState(({ resetId }) => ({ resetId: resetId + 1 }))
}
render() { render() {
const { resetId } = this.state
return ( return (
<Modal <Modal
name={MODAL_SEND} name={MODAL_SEND}
onHide={this.handleReset} preventBackdropClick
render={({ data, onClose }) => ( render={({ data, onClose }) => (
<SendModalBody <SendModalBody {...this.props} initialAccount={data && data.account} onClose={onClose} />
key={resetId}
{...this.props}
initialAccount={data && data.account}
onClose={onClose}
/>
)} )}
/> />
) )

2
static/i18n/en/send.yml

@ -31,3 +31,5 @@ steps:
error: error:
title: Transaction error title: Transaction error
cta: Retry operation cta: Retry operation
pending:
title: Broadcasting transaction...

Loading…
Cancel
Save