mirror of
https://github.com/trezor/trezor-wallet
synced 2024-11-30 12:18:09 +00:00
add import tool
This commit is contained in:
parent
4582538917
commit
8dd3f95f1f
@ -1,8 +1,13 @@
|
|||||||
/* @flow */
|
/* @flow */
|
||||||
|
|
||||||
import * as ACCOUNT from 'actions/constants/account';
|
import * as ACCOUNT from 'actions/constants/account';
|
||||||
import type { Action } from 'flowtype';
|
import * as NOTIFICATION from 'actions/constants/notification';
|
||||||
|
import type { Action, TrezorDevice, Network } 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 =
|
||||||
| {
|
| {
|
||||||
@ -18,3 +23,103 @@ 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 && 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: network.type,
|
||||||
|
nonce: account.nonce,
|
||||||
|
};
|
||||||
|
} else if (network.type === 'ripple') {
|
||||||
|
const response = await TrezorConnect.rippleGetAccountInfo({
|
||||||
|
account: {
|
||||||
|
descriptor: address,
|
||||||
|
},
|
||||||
|
coin: network.shortcut,
|
||||||
|
});
|
||||||
|
console.log(response);
|
||||||
|
|
||||||
|
// 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: network.type,
|
||||||
|
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,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -16,6 +16,7 @@ import type {
|
|||||||
Account,
|
Account,
|
||||||
} from 'flowtype';
|
} from 'flowtype';
|
||||||
import type { Discovery, State } from 'reducers/DiscoveryReducer';
|
import type { Discovery, State } from 'reducers/DiscoveryReducer';
|
||||||
|
import * as LocalStorageActions from 'actions/LocalStorageActions';
|
||||||
import * as BlockchainActions from './BlockchainActions';
|
import * as BlockchainActions from './BlockchainActions';
|
||||||
import * as EthereumDiscoveryActions from './ethereum/DiscoveryActions';
|
import * as EthereumDiscoveryActions from './ethereum/DiscoveryActions';
|
||||||
import * as RippleDiscoveryActions from './ripple/DiscoveryActions';
|
import * as RippleDiscoveryActions from './ripple/DiscoveryActions';
|
||||||
@ -120,6 +121,7 @@ const start = (device: TrezorDevice, network: string, ignoreCompleted?: boolean)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!discoveryProcess) {
|
if (!discoveryProcess) {
|
||||||
|
dispatch(addImportedAccounts());
|
||||||
dispatch(begin(device, network));
|
dispatch(begin(device, network));
|
||||||
} else if (discoveryProcess.completed && !ignoreCompleted) {
|
} else if (discoveryProcess.completed && !ignoreCompleted) {
|
||||||
dispatch({
|
dispatch({
|
||||||
@ -380,3 +382,17 @@ export const addAccount = (): ThunkAction => (dispatch: Dispatch, getState: GetS
|
|||||||
if (!selected) return;
|
if (!selected) return;
|
||||||
dispatch(start(selected, getState().router.location.state.network, true));
|
dispatch(start(selected, getState().router.location.state.network, true));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const addImportedAccounts = (): ThunkAction => (dispatch: Dispatch): void => {
|
||||||
|
// get imported accounts from local storage
|
||||||
|
const importedAccounts = LocalStorageActions.getImportedAccounts();
|
||||||
|
if (importedAccounts) {
|
||||||
|
// create each account
|
||||||
|
importedAccounts.forEach(account => {
|
||||||
|
dispatch({
|
||||||
|
type: ACCOUNT.CREATE,
|
||||||
|
payload: account,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -52,6 +52,7 @@ const { STORAGE_PATH } = storageUtils;
|
|||||||
const KEY_VERSION: string = `${STORAGE_PATH}version`;
|
const KEY_VERSION: string = `${STORAGE_PATH}version`;
|
||||||
const KEY_DEVICES: string = `${STORAGE_PATH}devices`;
|
const KEY_DEVICES: string = `${STORAGE_PATH}devices`;
|
||||||
const KEY_ACCOUNTS: string = `${STORAGE_PATH}accounts`;
|
const KEY_ACCOUNTS: string = `${STORAGE_PATH}accounts`;
|
||||||
|
const KEY_IMPORTED_ACCOUNTS: string = `${STORAGE_PATH}importedAccounts`;
|
||||||
const KEY_DISCOVERY: string = `${STORAGE_PATH}discovery`;
|
const KEY_DISCOVERY: string = `${STORAGE_PATH}discovery`;
|
||||||
const KEY_TOKENS: string = `${STORAGE_PATH}tokens`;
|
const KEY_TOKENS: string = `${STORAGE_PATH}tokens`;
|
||||||
const KEY_PENDING: string = `${STORAGE_PATH}pending`;
|
const KEY_PENDING: string = `${STORAGE_PATH}pending`;
|
||||||
@ -321,3 +322,20 @@ export const setLocalCurrency = (): ThunkAction => (
|
|||||||
const { localCurrency } = getState().wallet;
|
const { localCurrency } = getState().wallet;
|
||||||
storageUtils.set(TYPE, KEY_LOCAL_CURRENCY, JSON.stringify(localCurrency));
|
storageUtils.set(TYPE, KEY_LOCAL_CURRENCY, JSON.stringify(localCurrency));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const setImportedAccount = (account: Account): ThunkAction => (): void => {
|
||||||
|
const prevImportedAccounts: ?string = JSON.parse(storageUtils.get(TYPE, KEY_IMPORTED_ACCOUNTS));
|
||||||
|
let importedAccounts = [account];
|
||||||
|
if (prevImportedAccounts) {
|
||||||
|
importedAccounts = importedAccounts.concat(prevImportedAccounts);
|
||||||
|
}
|
||||||
|
storageUtils.set(TYPE, KEY_IMPORTED_ACCOUNTS, JSON.stringify(importedAccounts));
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getImportedAccounts = (): Array<Account> => {
|
||||||
|
const importedAccounts: ?string = storageUtils.get(TYPE, KEY_IMPORTED_ACCOUNTS);
|
||||||
|
if (importedAccounts) {
|
||||||
|
return JSON.parse(importedAccounts);
|
||||||
|
}
|
||||||
|
return importedAccounts;
|
||||||
|
};
|
||||||
|
@ -4,20 +4,24 @@ import * as React from 'react';
|
|||||||
import { Notification, Link } from 'trezor-ui-components';
|
import { Notification, Link } from 'trezor-ui-components';
|
||||||
import Bignumber from 'bignumber.js';
|
import Bignumber from 'bignumber.js';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
import { withRouter } from 'react-router-dom';
|
||||||
|
|
||||||
import l10nCommonMessages from 'views/common.messages';
|
import l10nCommonMessages from 'views/common.messages';
|
||||||
|
import { matchPath } from 'react-router';
|
||||||
|
import { getPattern } from 'support/routes';
|
||||||
import l10nMessages from './index.messages';
|
import l10nMessages from './index.messages';
|
||||||
import type { Props } from '../../index';
|
import type { Props } from '../../index';
|
||||||
|
|
||||||
export default (props: Props) => {
|
export default withRouter((props: Props) => {
|
||||||
const { selectedAccount } = props;
|
const { selectedAccount } = props;
|
||||||
const { account } = selectedAccount;
|
const { account } = selectedAccount;
|
||||||
const { location } = props.router;
|
const { location } = props.router;
|
||||||
const notifications: Array<Notification> = [];
|
const notifications: Array<Notification> = [];
|
||||||
|
|
||||||
if (!location || !selectedAccount || !account) return null;
|
if (!location) return null;
|
||||||
|
|
||||||
// Ripple minimum reserve notification
|
// Ripple minimum reserve notification
|
||||||
if (account.networkType === 'ripple') {
|
if (selectedAccount && account && account.networkType === 'ripple') {
|
||||||
const { reserve, balance } = account;
|
const { reserve, balance } = account;
|
||||||
const bigBalance = new Bignumber(balance);
|
const bigBalance = new Bignumber(balance);
|
||||||
const bigReserve = new Bignumber(reserve);
|
const bigReserve = new Bignumber(reserve);
|
||||||
@ -51,5 +55,28 @@ export default (props: Props) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Import tool notification
|
||||||
|
if (matchPath(location.pathname, { path: getPattern('wallet-import') })) {
|
||||||
|
notifications.push(
|
||||||
|
<Notification
|
||||||
|
key="import-warning"
|
||||||
|
type="warning"
|
||||||
|
title="Use at your own risk"
|
||||||
|
message="This is an advanced interface intended for developer use only. Never use this process unless you really know what you are doing."
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (account && account.imported) {
|
||||||
|
notifications.push(
|
||||||
|
<Notification
|
||||||
|
key="watch-only-info"
|
||||||
|
type="info"
|
||||||
|
title="The account is watch-only"
|
||||||
|
message="A watch-only account is a public address you’ve imported into your wallet, allowing the wallet to watch for outputs but not spend them."
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return <React.Fragment>{notifications}</React.Fragment>;
|
return <React.Fragment>{notifications}</React.Fragment>;
|
||||||
};
|
});
|
||||||
|
@ -39,6 +39,7 @@ export const FONT_SIZE = {
|
|||||||
H3: '1rem',
|
H3: '1rem',
|
||||||
H4: '0.8571rem',
|
H4: '0.8571rem',
|
||||||
COUNTER: '0.7857rem',
|
COUNTER: '0.7857rem',
|
||||||
|
BADGE: '0.7857rem',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const FONT_WEIGHT = {
|
export const FONT_WEIGHT = {
|
||||||
|
@ -95,7 +95,11 @@ const complete = (state: State, action: DiscoveryCompleteAction): State => {
|
|||||||
const accountCreate = (state: State, account: Account): State => {
|
const accountCreate = (state: State, account: Account): State => {
|
||||||
const index: number = findIndex(state, account.network, account.deviceState);
|
const index: number = findIndex(state, account.network, account.deviceState);
|
||||||
const newState: State = [...state];
|
const newState: State = [...state];
|
||||||
newState[index].accountIndex++;
|
// do not increment index when adding imported account
|
||||||
|
// imported accounts should not interfere with the index used in discovery proccess.
|
||||||
|
if (!account.imported) {
|
||||||
|
newState[index].accountIndex++;
|
||||||
|
}
|
||||||
return newState;
|
return newState;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -83,10 +83,24 @@ export const getSelectedAccount = (state: State): ?Account => {
|
|||||||
const locationState = state.router.location.state;
|
const locationState = state.router.location.state;
|
||||||
if (!device || !locationState.network || !locationState.account) return null;
|
if (!device || !locationState.network || !locationState.account) return null;
|
||||||
|
|
||||||
const index: number = parseInt(locationState.account, 10);
|
// imported account index has 'i' prefix
|
||||||
|
const isImported = /^i\d+$/i.test(locationState.account);
|
||||||
|
const index: number = isImported
|
||||||
|
? parseInt(locationState.account.substr(1), 10)
|
||||||
|
: parseInt(locationState.account, 10);
|
||||||
|
|
||||||
|
if (isImported) {
|
||||||
|
return state.accounts.find(
|
||||||
|
a =>
|
||||||
|
a.imported === true &&
|
||||||
|
a.deviceState === device.state &&
|
||||||
|
a.index === index &&
|
||||||
|
a.network === locationState.network
|
||||||
|
);
|
||||||
|
}
|
||||||
return state.accounts.find(
|
return state.accounts.find(
|
||||||
a =>
|
a =>
|
||||||
|
a.imported === false &&
|
||||||
a.deviceState === device.state &&
|
a.deviceState === device.state &&
|
||||||
a.index === index &&
|
a.index === index &&
|
||||||
a.network === locationState.network
|
a.network === locationState.network
|
||||||
|
@ -23,9 +23,9 @@ export const routes: Array<Route> = [
|
|||||||
fields: ['bridge'],
|
fields: ['bridge'],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'landing-import',
|
name: 'wallet-import',
|
||||||
pattern: '/import',
|
pattern: '/device/:device/import',
|
||||||
fields: ['import'],
|
fields: ['device', 'import'],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'wallet-settings',
|
name: 'wallet-settings',
|
||||||
|
@ -1,28 +0,0 @@
|
|||||||
/* @flow */
|
|
||||||
|
|
||||||
import React from 'react';
|
|
||||||
import styled from 'styled-components';
|
|
||||||
import { Button, Link, Icon, H5, icons, colors } from 'trezor-ui-components';
|
|
||||||
import LandingWrapper from 'views/Landing/components/LandingWrapper';
|
|
||||||
|
|
||||||
const Wrapper = styled.div`
|
|
||||||
width: 100%;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const Import = () => (
|
|
||||||
<LandingWrapper>
|
|
||||||
<Wrapper>
|
|
||||||
<Icon size={60} color={colors.WARNING_PRIMARY} icon={icons.WARNING} />
|
|
||||||
<H5>Import tool is under construction</H5>
|
|
||||||
<Link to="/">
|
|
||||||
<Button>Take me back</Button>
|
|
||||||
</Link>
|
|
||||||
</Wrapper>
|
|
||||||
</LandingWrapper>
|
|
||||||
);
|
|
||||||
|
|
||||||
export default Import;
|
|
@ -28,8 +28,6 @@ const Text = styled.span`
|
|||||||
const RowAccountWrapper = styled.div`
|
const RowAccountWrapper = styled.div`
|
||||||
width: 100%;
|
width: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
|
||||||
align-items: flex-start;
|
|
||||||
padding: ${LEFT_NAVIGATION_ROW.PADDING};
|
padding: ${LEFT_NAVIGATION_ROW.PADDING};
|
||||||
font-size: ${FONT_SIZE.BASE};
|
font-size: ${FONT_SIZE.BASE};
|
||||||
color: ${colors.TEXT_PRIMARY};
|
color: ${colors.TEXT_PRIMARY};
|
||||||
@ -86,6 +84,21 @@ const DiscoveryLoadingText = styled.span`
|
|||||||
margin-left: 14px;
|
margin-left: 14px;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const Col = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex: 1;
|
||||||
|
flex-direction: column;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const Badge = styled.div`
|
||||||
|
padding: 4px 8px;
|
||||||
|
background: lightslategray;
|
||||||
|
color: white;
|
||||||
|
font-size: ${FONT_SIZE.BADGE};
|
||||||
|
border-radius: 3px;
|
||||||
|
align-self: flex-end;
|
||||||
|
`;
|
||||||
|
|
||||||
// TODO: Refactorize deviceStatus & selectedAccounts
|
// TODO: Refactorize deviceStatus & selectedAccounts
|
||||||
const AccountMenu = (props: Props) => {
|
const AccountMenu = (props: Props) => {
|
||||||
const selected = props.wallet.selectedDevice;
|
const selected = props.wallet.selectedDevice;
|
||||||
@ -105,7 +118,12 @@ const AccountMenu = (props: Props) => {
|
|||||||
|
|
||||||
const selectedAccounts = deviceAccounts.map((account, i) => {
|
const selectedAccounts = deviceAccounts.map((account, i) => {
|
||||||
// const url: string = `${baseUrl}/network/${location.state.network}/account/${i}`;
|
// const url: string = `${baseUrl}/network/${location.state.network}/account/${i}`;
|
||||||
const url: string = location.pathname.replace(/account+\/([0-9]*)/, `account/${i}`);
|
let url: string;
|
||||||
|
if (account.imported) {
|
||||||
|
url = location.pathname.replace(/account+\/(i?[0-9]*)/, `account/i${account.index}`);
|
||||||
|
} else {
|
||||||
|
url = location.pathname.replace(/account+\/(i?[0-9]*)/, `account/${account.index}`);
|
||||||
|
}
|
||||||
|
|
||||||
let balance: ?string = null;
|
let balance: ?string = null;
|
||||||
const fiatRates = props.fiat.find(f => f.network === network.shortcut);
|
const fiatRates = props.fiat.find(f => f.network === network.shortcut);
|
||||||
@ -125,37 +143,38 @@ const AccountMenu = (props: Props) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const urlAccountIndex = parseInt(props.router.location.state.account, 10);
|
|
||||||
return (
|
return (
|
||||||
<NavLink to={url} key={account.index}>
|
<NavLink to={url} key={url}>
|
||||||
<Row column>
|
<Row column>
|
||||||
<RowAccountWrapper
|
<RowAccountWrapper isSelected={location.pathname === url} borderTop={i === 0}>
|
||||||
isSelected={urlAccountIndex === account.index}
|
<Col>
|
||||||
borderTop={account.index === 0}
|
<FormattedMessage
|
||||||
>
|
{...(account.imported
|
||||||
<FormattedMessage
|
? l10nCommonMessages.TR_IMPORTED_ACCOUNT_HASH
|
||||||
{...l10nCommonMessages.TR_ACCOUNT_HASH}
|
: l10nCommonMessages.TR_ACCOUNT_HASH)}
|
||||||
values={{ number: account.index + 1 }}
|
values={{ number: account.index + 1 }}
|
||||||
/>
|
/>
|
||||||
{balance && !props.wallet.hideBalance && (
|
{balance && !props.wallet.hideBalance && (
|
||||||
<Text>
|
<Text>
|
||||||
{balance}
|
{balance}
|
||||||
{fiatRates && (
|
{fiatRates && (
|
||||||
<FormattedNumber
|
<FormattedNumber
|
||||||
currency={localCurrency}
|
currency={localCurrency}
|
||||||
value={fiat}
|
value={fiat}
|
||||||
minimumFractionDigits={2}
|
minimumFractionDigits={2}
|
||||||
// eslint-disable-next-line react/style-prop-object
|
// eslint-disable-next-line react/style-prop-object
|
||||||
style="currency"
|
style="currency"
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
{!balance && (
|
{!balance && (
|
||||||
<Text>
|
<Text>
|
||||||
<FormattedMessage {...l10nMessages.TR_LOADING_DOT_DOT_DOT} />
|
<FormattedMessage {...l10nMessages.TR_LOADING_DOT_DOT_DOT} />
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
|
</Col>
|
||||||
|
<Col>{account.imported && <Badge>watch-only</Badge>}</Col>
|
||||||
</RowAccountWrapper>
|
</RowAccountWrapper>
|
||||||
</Row>
|
</Row>
|
||||||
</NavLink>
|
</NavLink>
|
||||||
|
@ -2,19 +2,20 @@
|
|||||||
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 RouterActions from 'actions/RouterActions';
|
import * as AccountsAction from 'actions/AccountsActions';
|
||||||
|
|
||||||
import type { MapStateToProps, MapDispatchToProps } from 'react-redux';
|
import type { MapStateToProps, MapDispatchToProps } from 'react-redux';
|
||||||
import type { State, Dispatch } from 'flowtype';
|
import type { Device, State, Dispatch } from 'flowtype';
|
||||||
import ImportView from './index';
|
import ImportView from './index';
|
||||||
|
|
||||||
export type StateProps = {
|
export type StateProps = {
|
||||||
transport: $ElementType<$ElementType<State, 'connect'>, 'transport'>,
|
transport: $ElementType<$ElementType<State, 'connect'>, 'transport'>,
|
||||||
|
device: ?Device,
|
||||||
children?: React.Node,
|
children?: React.Node,
|
||||||
};
|
};
|
||||||
|
|
||||||
type DispatchProps = {
|
type DispatchProps = {
|
||||||
selectFirstAvailableDevice: typeof RouterActions.selectFirstAvailableDevice,
|
importAddress: typeof AccountsAction.importAddress,
|
||||||
};
|
};
|
||||||
|
|
||||||
type OwnProps = {};
|
type OwnProps = {};
|
||||||
@ -24,16 +25,14 @@ export type Props = StateProps & DispatchProps;
|
|||||||
const mapStateToProps: MapStateToProps<State, OwnProps, StateProps> = (
|
const mapStateToProps: MapStateToProps<State, OwnProps, StateProps> = (
|
||||||
state: State
|
state: State
|
||||||
): StateProps => ({
|
): StateProps => ({
|
||||||
transport: state.connect.transport,
|
config: state.localStorage.config,
|
||||||
|
device: state.wallet.selectedDevice,
|
||||||
});
|
});
|
||||||
|
|
||||||
const mapDispatchToProps: MapDispatchToProps<Dispatch, OwnProps, DispatchProps> = (
|
const mapDispatchToProps: MapDispatchToProps<Dispatch, OwnProps, DispatchProps> = (
|
||||||
dispatch: Dispatch
|
dispatch: Dispatch
|
||||||
): DispatchProps => ({
|
): DispatchProps => ({
|
||||||
selectFirstAvailableDevice: bindActionCreators(
|
importAddress: bindActionCreators(AccountsAction.importAddress, dispatch),
|
||||||
RouterActions.selectFirstAvailableDevice,
|
|
||||||
dispatch
|
|
||||||
),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export default connect(
|
export default connect(
|
99
src/views/Wallet/views/Import/index.js
Normal file
99
src/views/Wallet/views/Import/index.js
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
/* @flow */
|
||||||
|
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
import { Select, Button, Input, Link, colors } from 'trezor-ui-components';
|
||||||
|
import l10nCommonMessages from 'views/common.messages';
|
||||||
|
import type { Props } from './Container';
|
||||||
|
|
||||||
|
const Wrapper = styled.div`
|
||||||
|
text-align: left;
|
||||||
|
flex-direction: column;
|
||||||
|
display: flex;
|
||||||
|
padding: 24px;
|
||||||
|
min-width: 300px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StyledSelect = styled(Select)`
|
||||||
|
min-width: 100px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const InputRow = styled.div`
|
||||||
|
margin-bottom: 16px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const Label = styled.div`
|
||||||
|
color: ${colors.TEXT_SECONDARY};
|
||||||
|
padding-bottom: 10px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const ButtonActions = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
|
||||||
|
justify-content: flex-end;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const ButtonWrapper = styled.div`
|
||||||
|
& + & {
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const Import = (props: Props) => {
|
||||||
|
const [selectedNetwork, setSelectedNetwork] = useState(null);
|
||||||
|
const [address, setAddress] = useState('');
|
||||||
|
|
||||||
|
const { networks } = props.config;
|
||||||
|
return (
|
||||||
|
// <LandingWrapper>
|
||||||
|
<Wrapper>
|
||||||
|
<InputRow>
|
||||||
|
<Label>Select network</Label>
|
||||||
|
<StyledSelect
|
||||||
|
value={selectedNetwork}
|
||||||
|
options={networks
|
||||||
|
.sort((a, b) => a.shortcut > b.shortcut)
|
||||||
|
.map(net => ({
|
||||||
|
label: net.shortcut,
|
||||||
|
value: net,
|
||||||
|
}))}
|
||||||
|
onChange={option => setSelectedNetwork(option)}
|
||||||
|
/>
|
||||||
|
</InputRow>
|
||||||
|
|
||||||
|
<InputRow>
|
||||||
|
<Input
|
||||||
|
topLabel="Address"
|
||||||
|
name="cryptoAddress"
|
||||||
|
value={address}
|
||||||
|
onChange={e => setAddress(e.target.value)}
|
||||||
|
type="text"
|
||||||
|
/>
|
||||||
|
</InputRow>
|
||||||
|
<ButtonActions>
|
||||||
|
<ButtonWrapper>
|
||||||
|
<Link to="/">
|
||||||
|
<Button isWhite>
|
||||||
|
<FormattedMessage {...l10nCommonMessages.TR_CLOSE} />
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</ButtonWrapper>
|
||||||
|
|
||||||
|
<ButtonWrapper>
|
||||||
|
<Button
|
||||||
|
isDisabled={!selectedNetwork || address === ''}
|
||||||
|
onClick={() =>
|
||||||
|
props.importAddress(address, selectedNetwork.value, props.device)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Import
|
||||||
|
</Button>
|
||||||
|
</ButtonWrapper>
|
||||||
|
</ButtonActions>
|
||||||
|
</Wrapper>
|
||||||
|
// </LandingWrapper>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
export default Import;
|
@ -16,6 +16,11 @@ const definedMessages: Messages = defineMessages({
|
|||||||
defaultMessage: 'Account #{number}',
|
defaultMessage: 'Account #{number}',
|
||||||
description: 'Used in auto-generated account label',
|
description: 'Used in auto-generated account label',
|
||||||
},
|
},
|
||||||
|
TR_IMPORTED_ACCOUNT_HASH: {
|
||||||
|
id: 'TR_IMPORTED_ACCOUNT_HASH',
|
||||||
|
defaultMessage: 'Imported account #{number}',
|
||||||
|
description: 'Used in auto-generated label for imported accounts',
|
||||||
|
},
|
||||||
TR_CLEAR: {
|
TR_CLEAR: {
|
||||||
id: 'TR_CLEAR',
|
id: 'TR_CLEAR',
|
||||||
defaultMessage: 'Clear',
|
defaultMessage: 'Clear',
|
||||||
|
@ -14,7 +14,6 @@ import { getPattern } from 'support/routes';
|
|||||||
// landing views
|
// landing views
|
||||||
import RootView from 'views/Landing/views/Root/Container';
|
import RootView from 'views/Landing/views/Root/Container';
|
||||||
import InstallBridge from 'views/Landing/views/InstallBridge/Container';
|
import InstallBridge from 'views/Landing/views/InstallBridge/Container';
|
||||||
import ImportView from 'views/Landing/views/Import/Container';
|
|
||||||
|
|
||||||
// wallet views
|
// wallet views
|
||||||
import WalletContainer from 'views/Wallet';
|
import WalletContainer from 'views/Wallet';
|
||||||
@ -23,6 +22,7 @@ import AccountSend from 'views/Wallet/views/Account/Send';
|
|||||||
import AccountReceive from 'views/Wallet/views/Account/Receive';
|
import AccountReceive from 'views/Wallet/views/Account/Receive';
|
||||||
import AccountSignVerify from 'views/Wallet/views/Account/SignVerify/Container';
|
import AccountSignVerify from 'views/Wallet/views/Account/SignVerify/Container';
|
||||||
|
|
||||||
|
import WalletImport from 'views/Wallet/views/Import/Container';
|
||||||
import WalletDashboard from 'views/Wallet/views/Dashboard/Container';
|
import WalletDashboard from 'views/Wallet/views/Dashboard/Container';
|
||||||
import WalletDeviceSettings from 'views/Wallet/views/DeviceSettings';
|
import WalletDeviceSettings from 'views/Wallet/views/DeviceSettings';
|
||||||
import WalletSettings from 'views/Wallet/views/WalletSettings/Container';
|
import WalletSettings from 'views/Wallet/views/WalletSettings/Container';
|
||||||
@ -44,11 +44,16 @@ const App = () => (
|
|||||||
<Route exact path={getPattern('landing-home')} component={RootView} />
|
<Route exact path={getPattern('landing-home')} component={RootView} />
|
||||||
<Route exact path={getPattern('landing-version')} component={Version} />
|
<Route exact path={getPattern('landing-version')} component={Version} />
|
||||||
<Route exact path={getPattern('landing-bridge')} component={InstallBridge} />
|
<Route exact path={getPattern('landing-bridge')} component={InstallBridge} />
|
||||||
<Route exact path={getPattern('landing-import')} component={ImportView} />
|
|
||||||
<Route>
|
<Route>
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<ImagesPreloader />
|
<ImagesPreloader />
|
||||||
<WalletContainer>
|
<WalletContainer>
|
||||||
|
<Route
|
||||||
|
exact
|
||||||
|
path={getPattern('wallet-import')}
|
||||||
|
component={WalletImport}
|
||||||
|
/>
|
||||||
|
|
||||||
<Route
|
<Route
|
||||||
exact
|
exact
|
||||||
path={getPattern('wallet-settings')}
|
path={getPattern('wallet-settings')}
|
||||||
|
Loading…
Reference in New Issue
Block a user