|
@ -1,6 +1,5 @@ |
|
|
// @flow
|
|
|
// @flow
|
|
|
|
|
|
|
|
|
import invariant from 'invariant' |
|
|
|
|
|
import React, { PureComponent } from 'react' |
|
|
import React, { PureComponent } from 'react' |
|
|
import { compose } from 'redux' |
|
|
import { compose } from 'redux' |
|
|
import { connect } from 'react-redux' |
|
|
import { connect } from 'react-redux' |
|
@ -13,22 +12,28 @@ import type { Currency, Account } from '@ledgerhq/live-common/lib/types' |
|
|
|
|
|
|
|
|
import { MODAL_ADD_ACCOUNTS } from 'config/constants' |
|
|
import { MODAL_ADD_ACCOUNTS } from 'config/constants' |
|
|
import type { T, Device } from 'types/common' |
|
|
import type { T, Device } from 'types/common' |
|
|
|
|
|
import type { StepProps as DefaultStepProps, Step } from 'components/base/Stepper' |
|
|
|
|
|
|
|
|
|
|
|
import { idleCallback } from 'helpers/promise' |
|
|
import { getCurrentDevice } from 'reducers/devices' |
|
|
import { getCurrentDevice } from 'reducers/devices' |
|
|
import { accountsSelector } from 'reducers/accounts' |
|
|
import { accountsSelector } from 'reducers/accounts' |
|
|
import { addAccount } from 'actions/accounts' |
|
|
import { addAccount } from 'actions/accounts' |
|
|
import { closeModal } from 'reducers/modals' |
|
|
import { closeModal } from 'reducers/modals' |
|
|
|
|
|
|
|
|
import Modal, { ModalContent, ModalTitle, ModalFooter, ModalBody } from 'components/base/Modal' |
|
|
import Modal from 'components/base/Modal' |
|
|
import Box from 'components/base/Box' |
|
|
import Stepper from 'components/base/Stepper' |
|
|
import Breadcrumb from 'components/Breadcrumb' |
|
|
|
|
|
|
|
|
|
|
|
import StepChooseCurrency, { StepChooseCurrencyFooter } from './steps/01-step-choose-currency' |
|
|
import StepChooseCurrency, { StepChooseCurrencyFooter } from './steps/01-step-choose-currency' |
|
|
import StepConnectDevice, { StepConnectDeviceFooter } from './steps/02-step-connect-device' |
|
|
import StepConnectDevice, { StepConnectDeviceFooter } from './steps/02-step-connect-device' |
|
|
import StepImport, { StepImportFooter } from './steps/03-step-import' |
|
|
import StepImport, { StepImportFooter } from './steps/03-step-import' |
|
|
import StepFinish from './steps/04-step-finish' |
|
|
import StepFinish from './steps/04-step-finish' |
|
|
|
|
|
|
|
|
const createSteps = ({ t }: { t: T }) => [ |
|
|
const createSteps = ({ t }: { t: T }) => { |
|
|
|
|
|
const onBack = ({ transitionTo, resetScanState }: StepProps) => { |
|
|
|
|
|
resetScanState() |
|
|
|
|
|
transitionTo('chooseCurrency') |
|
|
|
|
|
} |
|
|
|
|
|
return [ |
|
|
{ |
|
|
{ |
|
|
id: 'chooseCurrency', |
|
|
id: 'chooseCurrency', |
|
|
label: t('app:addAccounts.breadcrumb.informations'), |
|
|
label: t('app:addAccounts.breadcrumb.informations'), |
|
@ -42,7 +47,7 @@ const createSteps = ({ t }: { t: T }) => [ |
|
|
label: t('app:addAccounts.breadcrumb.connectDevice'), |
|
|
label: t('app:addAccounts.breadcrumb.connectDevice'), |
|
|
component: StepConnectDevice, |
|
|
component: StepConnectDevice, |
|
|
footer: StepConnectDeviceFooter, |
|
|
footer: StepConnectDeviceFooter, |
|
|
onBack: ({ transitionTo }: StepProps) => transitionTo('chooseCurrency'), |
|
|
onBack, |
|
|
hideFooter: false, |
|
|
hideFooter: false, |
|
|
}, |
|
|
}, |
|
|
{ |
|
|
{ |
|
@ -50,7 +55,7 @@ const createSteps = ({ t }: { t: T }) => [ |
|
|
label: t('app:addAccounts.breadcrumb.import'), |
|
|
label: t('app:addAccounts.breadcrumb.import'), |
|
|
component: StepImport, |
|
|
component: StepImport, |
|
|
footer: StepImportFooter, |
|
|
footer: StepImportFooter, |
|
|
onBack: ({ transitionTo }: StepProps) => transitionTo('chooseCurrency'), |
|
|
onBack, |
|
|
hideFooter: false, |
|
|
hideFooter: false, |
|
|
}, |
|
|
}, |
|
|
{ |
|
|
{ |
|
@ -62,10 +67,11 @@ const createSteps = ({ t }: { t: T }) => [ |
|
|
hideFooter: true, |
|
|
hideFooter: true, |
|
|
}, |
|
|
}, |
|
|
] |
|
|
] |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
type Props = { |
|
|
type Props = { |
|
|
t: T, |
|
|
t: T, |
|
|
currentDevice: ?Device, |
|
|
device: ?Device, |
|
|
existingAccounts: Account[], |
|
|
existingAccounts: Account[], |
|
|
closeModal: string => void, |
|
|
closeModal: string => void, |
|
|
addAccount: Account => void, |
|
|
addAccount: Account => void, |
|
@ -75,37 +81,39 @@ type StepId = 'chooseCurrency' | 'connectDevice' | 'import' | 'finish' |
|
|
type ScanStatus = 'idle' | 'scanning' | 'error' | 'finished' |
|
|
type ScanStatus = 'idle' | 'scanning' | 'error' | 'finished' |
|
|
|
|
|
|
|
|
type State = { |
|
|
type State = { |
|
|
stepId: StepId, |
|
|
// TODO: I'm sure there will be always StepId and ScanStatus given,
|
|
|
|
|
|
// but I struggle making flow understand it. So I put string as fallback
|
|
|
|
|
|
stepId: StepId | string, |
|
|
|
|
|
scanStatus: ScanStatus | string, |
|
|
|
|
|
|
|
|
isAppOpened: boolean, |
|
|
isAppOpened: boolean, |
|
|
currency: ?Currency, |
|
|
currency: ?Currency, |
|
|
|
|
|
|
|
|
// scan process
|
|
|
|
|
|
scannedAccounts: Account[], |
|
|
scannedAccounts: Account[], |
|
|
checkedAccountsIds: string[], |
|
|
checkedAccountsIds: string[], |
|
|
scanStatus: ScanStatus, |
|
|
|
|
|
err: ?Error, |
|
|
err: ?Error, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export type StepProps = { |
|
|
export type StepProps = DefaultStepProps & { |
|
|
t: T, |
|
|
t: T, |
|
|
currency: ?Currency, |
|
|
currency: ?Currency, |
|
|
currentDevice: ?Device, |
|
|
device: ?Device, |
|
|
isAppOpened: boolean, |
|
|
isAppOpened: boolean, |
|
|
transitionTo: StepId => void, |
|
|
|
|
|
setState: any => void, |
|
|
|
|
|
onClickAdd: void => Promise<void>, |
|
|
|
|
|
onCloseModal: void => void, |
|
|
|
|
|
|
|
|
|
|
|
// scan process
|
|
|
|
|
|
scannedAccounts: Account[], |
|
|
scannedAccounts: Account[], |
|
|
existingAccounts: Account[], |
|
|
existingAccounts: Account[], |
|
|
checkedAccountsIds: string[], |
|
|
checkedAccountsIds: string[], |
|
|
scanStatus: ScanStatus, |
|
|
scanStatus: ScanStatus, |
|
|
err: ?Error, |
|
|
err: ?Error, |
|
|
|
|
|
onClickAdd: void => Promise<void>, |
|
|
|
|
|
onCloseModal: void => void, |
|
|
|
|
|
resetScanState: void => void, |
|
|
|
|
|
setCurrency: (?Currency) => void, |
|
|
|
|
|
setAppOpened: boolean => void, |
|
|
|
|
|
setScanStatus: (ScanStatus, ?Error) => string, |
|
|
|
|
|
setScannedAccounts: ({ scannedAccounts?: Account[], checkedAccountsIds?: string[] }) => void, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const mapStateToProps = createStructuredSelector({ |
|
|
const mapStateToProps = createStructuredSelector({ |
|
|
currentDevice: getCurrentDevice, |
|
|
device: getCurrentDevice, |
|
|
existingAccounts: accountsSelector, |
|
|
existingAccounts: accountsSelector, |
|
|
}) |
|
|
}) |
|
|
|
|
|
|
|
@ -126,18 +134,7 @@ const INITIAL_STATE = { |
|
|
|
|
|
|
|
|
class AddAccounts extends PureComponent<Props, State> { |
|
|
class AddAccounts extends PureComponent<Props, State> { |
|
|
state = INITIAL_STATE |
|
|
state = INITIAL_STATE |
|
|
STEPS = createSteps({ |
|
|
STEPS = createSteps({ t: this.props.t }) |
|
|
t: this.props.t, |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
|
|
|
transitionTo = stepId => { |
|
|
|
|
|
const { currency } = this.state |
|
|
|
|
|
let nextState = { stepId } |
|
|
|
|
|
if (stepId === 'chooseCurrency') { |
|
|
|
|
|
nextState = { ...INITIAL_STATE, currency } |
|
|
|
|
|
} |
|
|
|
|
|
this.setState(nextState) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleClickAdd = async () => { |
|
|
handleClickAdd = async () => { |
|
|
const { addAccount } = this.props |
|
|
const { addAccount } = this.props |
|
@ -151,16 +148,43 @@ class AddAccounts extends PureComponent<Props, State> { |
|
|
await idleCallback() |
|
|
await idleCallback() |
|
|
addAccount(accountsToAdd[i]) |
|
|
addAccount(accountsToAdd[i]) |
|
|
} |
|
|
} |
|
|
this.transitionTo('finish') |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
handleCloseModal = () => { |
|
|
handleCloseModal = () => this.props.closeModal(MODAL_ADD_ACCOUNTS) |
|
|
const { closeModal } = this.props |
|
|
handleStepChange = (step: Step) => this.setState({ stepId: step.id }) |
|
|
closeModal(MODAL_ADD_ACCOUNTS) |
|
|
|
|
|
|
|
|
handleSetCurrency = (currency: ?Currency) => this.setState({ currency }) |
|
|
|
|
|
|
|
|
|
|
|
handleSetScanStatus = (scanStatus: string, err: ?Error = null) => { |
|
|
|
|
|
this.setState({ scanStatus, err }) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleSetScannedAccounts = ({ |
|
|
|
|
|
checkedAccountsIds, |
|
|
|
|
|
scannedAccounts, |
|
|
|
|
|
}: { |
|
|
|
|
|
checkedAccountsIds: string[], |
|
|
|
|
|
scannedAccounts: Account[], |
|
|
|
|
|
}) => { |
|
|
|
|
|
this.setState({ |
|
|
|
|
|
...(checkedAccountsIds ? { checkedAccountsIds } : {}), |
|
|
|
|
|
...(scannedAccounts ? { scannedAccounts } : {}), |
|
|
|
|
|
}) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleResetScanState = () => { |
|
|
|
|
|
this.setState({ |
|
|
|
|
|
scanStatus: 'idle', |
|
|
|
|
|
err: null, |
|
|
|
|
|
scannedAccounts: [], |
|
|
|
|
|
checkedAccountsIds: [], |
|
|
|
|
|
}) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleSetAppOpened = (isAppOpened: boolean) => this.setState({ isAppOpened }) |
|
|
|
|
|
|
|
|
render() { |
|
|
render() { |
|
|
const { t, currentDevice, existingAccounts } = this.props |
|
|
const { t, device, existingAccounts } = this.props |
|
|
const { |
|
|
const { |
|
|
stepId, |
|
|
stepId, |
|
|
currency, |
|
|
currency, |
|
@ -171,17 +195,9 @@ class AddAccounts extends PureComponent<Props, State> { |
|
|
err, |
|
|
err, |
|
|
} = this.state |
|
|
} = this.state |
|
|
|
|
|
|
|
|
const stepIndex = this.STEPS.findIndex(s => s.id === stepId) |
|
|
const addtionnalProps = { |
|
|
const step = this.STEPS[stepIndex] |
|
|
|
|
|
|
|
|
|
|
|
invariant(step, `AddAccountsModal: step ${stepId} doesn't exists`) |
|
|
|
|
|
|
|
|
|
|
|
const { component: StepComponent, footer: StepFooter, hideFooter, onBack } = step |
|
|
|
|
|
|
|
|
|
|
|
const stepProps: StepProps = { |
|
|
|
|
|
t, |
|
|
|
|
|
currency, |
|
|
currency, |
|
|
currentDevice, |
|
|
device, |
|
|
existingAccounts, |
|
|
existingAccounts, |
|
|
scannedAccounts, |
|
|
scannedAccounts, |
|
|
checkedAccountsIds, |
|
|
checkedAccountsIds, |
|
@ -190,8 +206,11 @@ class AddAccounts extends PureComponent<Props, State> { |
|
|
isAppOpened, |
|
|
isAppOpened, |
|
|
onClickAdd: this.handleClickAdd, |
|
|
onClickAdd: this.handleClickAdd, |
|
|
onCloseModal: this.handleCloseModal, |
|
|
onCloseModal: this.handleCloseModal, |
|
|
transitionTo: this.transitionTo, |
|
|
setScanStatus: this.handleSetScanStatus, |
|
|
setState: (...args) => this.setState(...args), |
|
|
setCurrency: this.handleSetCurrency, |
|
|
|
|
|
setScannedAccounts: this.handleSetScannedAccounts, |
|
|
|
|
|
resetScanState: this.handleResetScanState, |
|
|
|
|
|
setAppOpened: this.handleSetAppOpened, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return ( |
|
|
return ( |
|
@ -200,21 +219,16 @@ class AddAccounts extends PureComponent<Props, State> { |
|
|
refocusWhenChange={stepId} |
|
|
refocusWhenChange={stepId} |
|
|
onHide={() => this.setState({ ...INITIAL_STATE })} |
|
|
onHide={() => this.setState({ ...INITIAL_STATE })} |
|
|
render={({ onClose }) => ( |
|
|
render={({ onClose }) => ( |
|
|
<ModalBody onClose={onClose}> |
|
|
<Stepper |
|
|
|
|
|
title={t('app:addAccounts.title')} |
|
|
|
|
|
initialStepId="chooseCurrency" |
|
|
|
|
|
onStepChange={this.handleStepChange} |
|
|
|
|
|
onClose={onClose} |
|
|
|
|
|
steps={this.STEPS} |
|
|
|
|
|
{...addtionnalProps} |
|
|
|
|
|
> |
|
|
<SyncSkipUnderPriority priority={100} /> |
|
|
<SyncSkipUnderPriority priority={100} /> |
|
|
<ModalTitle onBack={onBack ? () => onBack(stepProps) : void 0}> |
|
|
</Stepper> |
|
|
{t('app:addAccounts.title')} |
|
|
|
|
|
</ModalTitle> |
|
|
|
|
|
<ModalContent> |
|
|
|
|
|
<Breadcrumb mb={6} currentStep={stepIndex} items={this.STEPS} /> |
|
|
|
|
|
<StepComponent {...stepProps} /> |
|
|
|
|
|
</ModalContent> |
|
|
|
|
|
{!hideFooter && ( |
|
|
|
|
|
<ModalFooter horizontal align="center" justify="flex-end" style={{ height: 80 }}> |
|
|
|
|
|
{StepFooter ? <StepFooter {...stepProps} /> : <Box />} |
|
|
|
|
|
</ModalFooter> |
|
|
|
|
|
)} |
|
|
|
|
|
</ModalBody> |
|
|
|
|
|
)} |
|
|
)} |
|
|
/> |
|
|
/> |
|
|
) |
|
|
) |
|
@ -228,7 +242,3 @@ export default compose( |
|
|
), |
|
|
), |
|
|
translate(), |
|
|
translate(), |
|
|
)(AddAccounts) |
|
|
)(AddAccounts) |
|
|
|
|
|
|
|
|
function idleCallback() { |
|
|
|
|
|
return new Promise(resolve => window.requestIdleCallback(resolve)) |
|
|
|
|
|
} |
|
|
|
|
|