mirror of
https://github.com/trezor/trezor-wallet
synced 2025-02-15 17:42:03 +00:00
create separate reducer ImportAccount
This commit is contained in:
parent
b91c78a09e
commit
0f6c9a70ed
@ -1,13 +1,8 @@
|
|||||||
/* @flow */
|
/* @flow */
|
||||||
|
|
||||||
import * as ACCOUNT from 'actions/constants/account';
|
import * as ACCOUNT from 'actions/constants/account';
|
||||||
import * as NOTIFICATION from 'actions/constants/notification';
|
import type { Action } from 'flowtype';
|
||||||
import type { Action, AsyncAction, TrezorDevice, Network, Dispatch, GetState } from 'flowtype';
|
|
||||||
import type { Account, State } from 'reducers/AccountsReducer';
|
import type { Account, State } from 'reducers/AccountsReducer';
|
||||||
import * as BlockchainActions from 'actions/ethereum/BlockchainActions';
|
|
||||||
import * as LocalStorageActions from 'actions/LocalStorageActions';
|
|
||||||
import TrezorConnect from 'trezor-connect';
|
|
||||||
import { toDecimalAmount } from 'utils/formatUtils';
|
|
||||||
|
|
||||||
export type AccountAction =
|
export type AccountAction =
|
||||||
| {
|
| {
|
||||||
@ -23,119 +18,3 @@ export const update = (account: Account): Action => ({
|
|||||||
type: ACCOUNT.UPDATE,
|
type: ACCOUNT.UPDATE,
|
||||||
payload: account,
|
payload: account,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const importAddress = (
|
|
||||||
address: string,
|
|
||||||
network: Network,
|
|
||||||
device: ?TrezorDevice
|
|
||||||
): AsyncAction => async (dispatch: Dispatch, getState: GetState): Promise<void> => {
|
|
||||||
if (!device) return;
|
|
||||||
|
|
||||||
let payload;
|
|
||||||
const index = getState().accounts.filter(
|
|
||||||
a =>
|
|
||||||
a.imported === true &&
|
|
||||||
a.network === network.shortcut &&
|
|
||||||
device &&
|
|
||||||
a.deviceState === device.state
|
|
||||||
).length;
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (network.type === 'ethereum') {
|
|
||||||
const account = await dispatch(
|
|
||||||
BlockchainActions.discoverAccount(device, address, network.shortcut)
|
|
||||||
);
|
|
||||||
|
|
||||||
const empty = account.nonce <= 0 && account.balance === '0';
|
|
||||||
payload = {
|
|
||||||
imported: true,
|
|
||||||
index,
|
|
||||||
network: network.shortcut,
|
|
||||||
deviceID: device.features ? device.features.device_id : '0',
|
|
||||||
deviceState: device.state || '0',
|
|
||||||
accountPath: account.path || [],
|
|
||||||
descriptor: account.descriptor,
|
|
||||||
|
|
||||||
balance: account.balance,
|
|
||||||
availableBalance: account.balance,
|
|
||||||
block: account.block,
|
|
||||||
transactions: account.transactions,
|
|
||||||
empty,
|
|
||||||
|
|
||||||
networkType: 'ethereum',
|
|
||||||
nonce: account.nonce,
|
|
||||||
};
|
|
||||||
dispatch({
|
|
||||||
type: ACCOUNT.CREATE,
|
|
||||||
payload,
|
|
||||||
});
|
|
||||||
dispatch(LocalStorageActions.setImportedAccount(payload));
|
|
||||||
dispatch({
|
|
||||||
type: NOTIFICATION.ADD,
|
|
||||||
payload: {
|
|
||||||
type: 'success',
|
|
||||||
title: 'The account has been successfully imported',
|
|
||||||
cancelable: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else if (network.type === 'ripple') {
|
|
||||||
const response = await TrezorConnect.rippleGetAccountInfo({
|
|
||||||
account: {
|
|
||||||
descriptor: address,
|
|
||||||
},
|
|
||||||
coin: network.shortcut,
|
|
||||||
});
|
|
||||||
|
|
||||||
// handle TREZOR response error
|
|
||||||
if (!response.success) {
|
|
||||||
throw new Error(response.payload.error);
|
|
||||||
}
|
|
||||||
|
|
||||||
const account = response.payload;
|
|
||||||
const empty = account.sequence <= 0 && account.balance === '0';
|
|
||||||
|
|
||||||
payload = {
|
|
||||||
imported: true,
|
|
||||||
index,
|
|
||||||
network: network.shortcut,
|
|
||||||
deviceID: device.features ? device.features.device_id : '0',
|
|
||||||
deviceState: device.state || '0',
|
|
||||||
accountPath: account.path || [],
|
|
||||||
descriptor: account.descriptor,
|
|
||||||
|
|
||||||
balance: toDecimalAmount(account.balance, network.decimals),
|
|
||||||
availableBalance: toDecimalAmount(account.availableBalance, network.decimals),
|
|
||||||
block: account.block,
|
|
||||||
transactions: account.transactions,
|
|
||||||
empty,
|
|
||||||
|
|
||||||
networkType: 'ripple',
|
|
||||||
sequence: account.sequence,
|
|
||||||
reserve: toDecimalAmount(account.reserve, network.decimals),
|
|
||||||
};
|
|
||||||
dispatch({
|
|
||||||
type: ACCOUNT.CREATE,
|
|
||||||
payload,
|
|
||||||
});
|
|
||||||
dispatch(LocalStorageActions.setImportedAccount(payload));
|
|
||||||
dispatch({
|
|
||||||
type: NOTIFICATION.ADD,
|
|
||||||
payload: {
|
|
||||||
type: 'success',
|
|
||||||
title: 'The account has been successfully imported',
|
|
||||||
cancelable: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
dispatch({
|
|
||||||
type: NOTIFICATION.ADD,
|
|
||||||
payload: {
|
|
||||||
type: 'error',
|
|
||||||
title: 'Import account error',
|
|
||||||
message: error.message,
|
|
||||||
cancelable: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
153
src/actions/ImportAccountActions.js
Normal file
153
src/actions/ImportAccountActions.js
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
/* @flow */
|
||||||
|
|
||||||
|
import * as ACCOUNT from 'actions/constants/account';
|
||||||
|
import * as IMPORT from 'actions/constants/importAccount';
|
||||||
|
import * as NOTIFICATION from 'actions/constants/notification';
|
||||||
|
import type { AsyncAction, TrezorDevice, Network, Dispatch, GetState } from 'flowtype';
|
||||||
|
import * as BlockchainActions from 'actions/ethereum/BlockchainActions';
|
||||||
|
import * as LocalStorageActions from 'actions/LocalStorageActions';
|
||||||
|
import TrezorConnect from 'trezor-connect';
|
||||||
|
import { toDecimalAmount } from 'utils/formatUtils';
|
||||||
|
|
||||||
|
export type ImportAccountAction =
|
||||||
|
| {
|
||||||
|
type: typeof IMPORT.START,
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: typeof IMPORT.SUCCESS,
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: typeof IMPORT.FAIL,
|
||||||
|
error: ?string,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const importAddress = (
|
||||||
|
address: string,
|
||||||
|
network: Network,
|
||||||
|
device: ?TrezorDevice
|
||||||
|
): AsyncAction => async (dispatch: Dispatch, getState: GetState): Promise<void> => {
|
||||||
|
if (!device) return;
|
||||||
|
|
||||||
|
dispatch({
|
||||||
|
type: IMPORT.START,
|
||||||
|
});
|
||||||
|
|
||||||
|
let payload;
|
||||||
|
const index = getState().accounts.filter(
|
||||||
|
a =>
|
||||||
|
a.imported === true &&
|
||||||
|
a.network === network.shortcut &&
|
||||||
|
device &&
|
||||||
|
a.deviceState === device.state
|
||||||
|
).length;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (network.type === 'ethereum') {
|
||||||
|
const account = await dispatch(
|
||||||
|
BlockchainActions.discoverAccount(device, address, network.shortcut)
|
||||||
|
);
|
||||||
|
|
||||||
|
const empty = account.nonce <= 0 && account.balance === '0';
|
||||||
|
payload = {
|
||||||
|
imported: true,
|
||||||
|
index,
|
||||||
|
network: network.shortcut,
|
||||||
|
deviceID: device.features ? device.features.device_id : '0',
|
||||||
|
deviceState: device.state || '0',
|
||||||
|
accountPath: account.path || [],
|
||||||
|
descriptor: account.descriptor,
|
||||||
|
|
||||||
|
balance: account.balance,
|
||||||
|
availableBalance: account.balance,
|
||||||
|
block: account.block,
|
||||||
|
transactions: account.transactions,
|
||||||
|
empty,
|
||||||
|
|
||||||
|
networkType: 'ethereum',
|
||||||
|
nonce: account.nonce,
|
||||||
|
};
|
||||||
|
dispatch({
|
||||||
|
type: ACCOUNT.CREATE,
|
||||||
|
payload,
|
||||||
|
});
|
||||||
|
dispatch({
|
||||||
|
type: IMPORT.SUCCESS,
|
||||||
|
});
|
||||||
|
dispatch(LocalStorageActions.setImportedAccount(payload));
|
||||||
|
dispatch({
|
||||||
|
type: NOTIFICATION.ADD,
|
||||||
|
payload: {
|
||||||
|
type: 'success',
|
||||||
|
title: 'The account has been successfully imported',
|
||||||
|
cancelable: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else if (network.type === 'ripple') {
|
||||||
|
const response = await TrezorConnect.rippleGetAccountInfo({
|
||||||
|
account: {
|
||||||
|
descriptor: address,
|
||||||
|
},
|
||||||
|
coin: network.shortcut,
|
||||||
|
});
|
||||||
|
|
||||||
|
// handle TREZOR response error
|
||||||
|
if (!response.success) {
|
||||||
|
throw new Error(response.payload.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
const account = response.payload;
|
||||||
|
const empty = account.sequence <= 0 && account.balance === '0';
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
imported: true,
|
||||||
|
index,
|
||||||
|
network: network.shortcut,
|
||||||
|
deviceID: device.features ? device.features.device_id : '0',
|
||||||
|
deviceState: device.state || '0',
|
||||||
|
accountPath: account.path || [],
|
||||||
|
descriptor: account.descriptor,
|
||||||
|
|
||||||
|
balance: toDecimalAmount(account.balance, network.decimals),
|
||||||
|
availableBalance: toDecimalAmount(account.availableBalance, network.decimals),
|
||||||
|
block: account.block,
|
||||||
|
transactions: account.transactions,
|
||||||
|
empty,
|
||||||
|
|
||||||
|
networkType: 'ripple',
|
||||||
|
sequence: account.sequence,
|
||||||
|
reserve: toDecimalAmount(account.reserve, network.decimals),
|
||||||
|
};
|
||||||
|
dispatch({
|
||||||
|
type: ACCOUNT.CREATE,
|
||||||
|
payload,
|
||||||
|
});
|
||||||
|
dispatch({
|
||||||
|
type: IMPORT.SUCCESS,
|
||||||
|
});
|
||||||
|
dispatch(LocalStorageActions.setImportedAccount(payload));
|
||||||
|
dispatch({
|
||||||
|
type: NOTIFICATION.ADD,
|
||||||
|
payload: {
|
||||||
|
type: 'success',
|
||||||
|
title: 'The account has been successfully imported',
|
||||||
|
cancelable: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
dispatch({
|
||||||
|
type: IMPORT.FAIL,
|
||||||
|
error: error.message,
|
||||||
|
});
|
||||||
|
|
||||||
|
dispatch({
|
||||||
|
type: NOTIFICATION.ADD,
|
||||||
|
payload: {
|
||||||
|
type: 'error',
|
||||||
|
title: 'Import account error',
|
||||||
|
message: error.message,
|
||||||
|
cancelable: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
5
src/actions/constants/importAccount.js
Normal file
5
src/actions/constants/importAccount.js
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
/* @flow */
|
||||||
|
|
||||||
|
export const START: 'import__account__start' = 'import__account__start';
|
||||||
|
export const SUCCESS: 'import__account__success' = 'import__account__success';
|
||||||
|
export const FAIL: 'import__account__fail' = 'import__account__fail';
|
@ -32,6 +32,7 @@ import type { TokenAction } from 'actions/TokenActions';
|
|||||||
import type { TrezorConnectAction } from 'actions/TrezorConnectActions';
|
import type { TrezorConnectAction } from 'actions/TrezorConnectActions';
|
||||||
import type { WalletAction } from 'actions/WalletActions';
|
import type { WalletAction } from 'actions/WalletActions';
|
||||||
import type { Web3Action } from 'actions/Web3Actions';
|
import type { Web3Action } from 'actions/Web3Actions';
|
||||||
|
import type { ImportAccountAction } from 'actions/ImportAccountActions';
|
||||||
import type { FiatRateAction } from 'services/CoingeckoService'; // this service has no action file, all is written inside one file
|
import type { FiatRateAction } from 'services/CoingeckoService'; // this service has no action file, all is written inside one file
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
@ -142,7 +143,8 @@ export type Action =
|
|||||||
| TrezorConnectAction
|
| TrezorConnectAction
|
||||||
| WalletAction
|
| WalletAction
|
||||||
| Web3Action
|
| Web3Action
|
||||||
| FiatRateAction;
|
| FiatRateAction
|
||||||
|
| ImportAccountAction;
|
||||||
|
|
||||||
export type State = ReducersState;
|
export type State = ReducersState;
|
||||||
|
|
||||||
|
43
src/reducers/ImportAccountReducer.js
Normal file
43
src/reducers/ImportAccountReducer.js
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
/* @flow */
|
||||||
|
|
||||||
|
import * as IMPORT from 'actions/constants/importAccount';
|
||||||
|
|
||||||
|
import type { Action } from 'flowtype';
|
||||||
|
|
||||||
|
export type ImportState = {
|
||||||
|
loading: boolean,
|
||||||
|
error: ?string,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const initialState: ImportState = {
|
||||||
|
loading: false,
|
||||||
|
error: null,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default (state: ImportState = initialState, action: Action): ImportState => {
|
||||||
|
switch (action.type) {
|
||||||
|
case IMPORT.START:
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
loading: true,
|
||||||
|
error: null,
|
||||||
|
};
|
||||||
|
|
||||||
|
case IMPORT.SUCCESS:
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
loading: false,
|
||||||
|
error: null,
|
||||||
|
};
|
||||||
|
|
||||||
|
case IMPORT.FAIL:
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
loading: false,
|
||||||
|
error: action.error,
|
||||||
|
};
|
||||||
|
|
||||||
|
default:
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
};
|
@ -10,6 +10,7 @@ import notifications from 'reducers/NotificationReducer';
|
|||||||
import modal from 'reducers/ModalReducer';
|
import modal from 'reducers/ModalReducer';
|
||||||
import web3 from 'reducers/Web3Reducer';
|
import web3 from 'reducers/Web3Reducer';
|
||||||
import accounts from 'reducers/AccountsReducer';
|
import accounts from 'reducers/AccountsReducer';
|
||||||
|
import importAccount from 'reducers/ImportAccountReducer';
|
||||||
import selectedAccount from 'reducers/SelectedAccountReducer';
|
import selectedAccount from 'reducers/SelectedAccountReducer';
|
||||||
import sendFormEthereum from 'reducers/SendFormEthereumReducer';
|
import sendFormEthereum from 'reducers/SendFormEthereumReducer';
|
||||||
import sendFormRipple from 'reducers/SendFormRippleReducer';
|
import sendFormRipple from 'reducers/SendFormRippleReducer';
|
||||||
@ -31,6 +32,7 @@ const reducers = {
|
|||||||
notifications,
|
notifications,
|
||||||
modal,
|
modal,
|
||||||
web3,
|
web3,
|
||||||
|
importAccount,
|
||||||
accounts,
|
accounts,
|
||||||
selectedAccount,
|
selectedAccount,
|
||||||
sendFormEthereum,
|
sendFormEthereum,
|
||||||
|
@ -88,7 +88,7 @@ const Col = styled.div`
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex: 1;
|
flex: 1;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
`
|
`;
|
||||||
|
|
||||||
const RightCol = styled(Col)`
|
const RightCol = styled(Col)`
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { bindActionCreators } from 'redux';
|
import { bindActionCreators } from 'redux';
|
||||||
import * as AccountsAction from 'actions/AccountsActions';
|
import * as ImportAccountActions from 'actions/ImportAccountActions';
|
||||||
|
|
||||||
import type { MapStateToProps, MapDispatchToProps } from 'react-redux';
|
import type { MapStateToProps, MapDispatchToProps } from 'react-redux';
|
||||||
import type { TrezorDevice, Config, State, Dispatch } from 'flowtype';
|
import type { TrezorDevice, Config, State, Dispatch } from 'flowtype';
|
||||||
@ -10,12 +10,13 @@ import ImportView from './index';
|
|||||||
|
|
||||||
export type StateProps = {
|
export type StateProps = {
|
||||||
device: ?TrezorDevice,
|
device: ?TrezorDevice,
|
||||||
|
importAccount: $ElementType<State, 'importAccount'>,
|
||||||
config: Config,
|
config: Config,
|
||||||
children?: React.Node,
|
children?: React.Node,
|
||||||
};
|
};
|
||||||
|
|
||||||
type DispatchProps = {
|
type DispatchProps = {
|
||||||
importAddress: typeof AccountsAction.importAddress,
|
importAddress: typeof ImportAccountActions.importAddress,
|
||||||
};
|
};
|
||||||
|
|
||||||
type OwnProps = {};
|
type OwnProps = {};
|
||||||
@ -27,12 +28,13 @@ const mapStateToProps: MapStateToProps<State, OwnProps, StateProps> = (
|
|||||||
): StateProps => ({
|
): StateProps => ({
|
||||||
config: state.localStorage.config,
|
config: state.localStorage.config,
|
||||||
device: state.wallet.selectedDevice,
|
device: state.wallet.selectedDevice,
|
||||||
|
importAccount: state.importAccount,
|
||||||
});
|
});
|
||||||
|
|
||||||
const mapDispatchToProps: MapDispatchToProps<Dispatch, OwnProps, DispatchProps> = (
|
const mapDispatchToProps: MapDispatchToProps<Dispatch, OwnProps, DispatchProps> = (
|
||||||
dispatch: Dispatch
|
dispatch: Dispatch
|
||||||
): DispatchProps => ({
|
): DispatchProps => ({
|
||||||
importAddress: bindActionCreators(AccountsAction.importAddress, dispatch),
|
importAddress: bindActionCreators(ImportAccountActions.importAddress, dispatch),
|
||||||
});
|
});
|
||||||
|
|
||||||
export default connect(
|
export default connect(
|
||||||
|
@ -83,7 +83,9 @@ const Import = (props: Props) => {
|
|||||||
|
|
||||||
<ButtonWrapper>
|
<ButtonWrapper>
|
||||||
<Button
|
<Button
|
||||||
isDisabled={!selectedNetwork || address === ''}
|
isDisabled={
|
||||||
|
!selectedNetwork || address === '' || props.importAccount.loading
|
||||||
|
}
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
props.importAddress(address, selectedNetwork.value, props.device)
|
props.importAddress(address, selectedNetwork.value, props.device)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user