feat: integrate wallet flow with heartbeat and other branches

This commit is contained in:
fragosti 2018-11-12 13:30:47 -08:00
parent a8a1ea92a6
commit 79f0324abc
6 changed files with 82 additions and 49 deletions

View File

@ -1,7 +1,7 @@
import * as _ from 'lodash'; import * as _ from 'lodash';
import * as React from 'react'; import * as React from 'react';
import { ProgressBar, TimedProgressBar } from '../components/timed_progress_bar'; import { TimedProgressBar } from '../components/timed_progress_bar';
import { TimeCounter } from '../components/time_counter'; import { TimeCounter } from '../components/time_counter';
import { Container } from '../components/ui/container'; import { Container } from '../components/ui/container';

View File

@ -2,7 +2,7 @@ import * as _ from 'lodash';
import * as React from 'react'; import * as React from 'react';
import { ColorOption } from '../style/theme'; import { ColorOption } from '../style/theme';
import { Account, AccountState, Network, StandardSlidingPanelContent } from '../types'; import { Account, AccountState, Network } from '../types';
import { MetaMaskLogo } from './meta_mask_logo'; import { MetaMaskLogo } from './meta_mask_logo';
import { PaymentMethodDropdown } from './payment_method_dropdown'; import { PaymentMethodDropdown } from './payment_method_dropdown';
@ -15,7 +15,8 @@ import { Text } from './ui/text';
export interface PaymentMethodProps { export interface PaymentMethodProps {
account: Account; account: Account;
network: Network; network: Network;
openStandardSlidingPanel: (content: StandardSlidingPanelContent) => void; onInstallWalletClick: () => void;
onUnlockWalletClick: () => void;
} }
export class PaymentMethod extends React.Component<PaymentMethodProps> { export class PaymentMethod extends React.Component<PaymentMethodProps> {
@ -80,7 +81,7 @@ export class PaymentMethod extends React.Component<PaymentMethodProps> {
case AccountState.Locked: case AccountState.Locked:
return ( return (
<WalletPrompt <WalletPrompt
onClick={this._openInstallWalletPanel} onClick={this.props.onUnlockWalletClick}
image={<Icon width={13} icon="lock" color={ColorOption.black} />} image={<Icon width={13} icon="lock" color={ColorOption.black} />}
> >
Please Unlock MetaMask Please Unlock MetaMask
@ -89,7 +90,7 @@ export class PaymentMethod extends React.Component<PaymentMethodProps> {
case AccountState.None: case AccountState.None:
return ( return (
<WalletPrompt <WalletPrompt
onClick={this._openInstallWalletPanel} onClick={this.props.onInstallWalletClick}
image={<MetaMaskLogo width={19} height={18} />} image={<MetaMaskLogo width={19} height={18} />}
> >
Install MetaMask Install MetaMask
@ -107,9 +108,6 @@ export class PaymentMethod extends React.Component<PaymentMethodProps> {
return null; return null;
} }
}; };
private readonly _openInstallWalletPanel = () => {
this.props.openStandardSlidingPanel(StandardSlidingPanelContent.InstallWallet);
};
} }
interface WalletPromptProps { interface WalletPromptProps {

View File

@ -91,12 +91,13 @@ export class ZeroExInstantProvider extends React.Component<ZeroExInstantProvider
} }
public componentDidMount(): void { public componentDidMount(): void {
const state = this._store.getState(); const state = this._store.getState();
const dispatch = this._store.dispatch;
// tslint:disable-next-line:no-floating-promises // tslint:disable-next-line:no-floating-promises
asyncData.fetchEthPriceAndDispatchToStore(this._store); asyncData.fetchEthPriceAndDispatchToStore(dispatch);
// fetch available assets if none are specified // fetch available assets if none are specified
if (_.isUndefined(state.availableAssets)) { if (_.isUndefined(state.availableAssets)) {
// tslint:disable-next-line:no-floating-promises // tslint:disable-next-line:no-floating-promises
asyncData.fetchAvailableAssetDatasAndDispatchToStore(this._store); asyncData.fetchAvailableAssetDatasAndDispatchToStore(state, dispatch);
} }
if (state.providerState.account.state !== AccountState.None) { if (state.providerState.account.state !== AccountState.None) {
this._accountUpdateHeartbeat = generateAccountHeartbeater({ this._accountUpdateHeartbeat = generateAccountHeartbeater({
@ -112,7 +113,7 @@ export class ZeroExInstantProvider extends React.Component<ZeroExInstantProvider
}); });
this._buyQuoteHeartbeat.start(BUY_QUOTE_UPDATE_INTERVAL_TIME_MS); this._buyQuoteHeartbeat.start(BUY_QUOTE_UPDATE_INTERVAL_TIME_MS);
// tslint:disable-next-line:no-floating-promises // tslint:disable-next-line:no-floating-promises
asyncData.fetchCurrentBuyQuoteAndDispatchToStore({ store: this._store, shouldSetPending: true }); asyncData.fetchCurrentBuyQuoteAndDispatchToStore(state, dispatch, true);
// warm up the gas price estimator cache just in case we can't // warm up the gas price estimator cache just in case we can't
// grab the gas price estimate when submitting the transaction // grab the gas price estimate when submitting the transaction
// tslint:disable-next-line:no-floating-promises // tslint:disable-next-line:no-floating-promises

View File

@ -4,34 +4,61 @@ import { Dispatch } from 'redux';
import { PaymentMethod } from '../components/payment_method'; import { PaymentMethod } from '../components/payment_method';
import { Action, actions } from '../redux/actions'; import { Action, actions } from '../redux/actions';
import { asyncData } from '../redux/async_data';
import { State } from '../redux/reducer'; import { State } from '../redux/reducer';
import { Account, Network, StandardSlidingPanelContent } from '../types'; import { Account, Network, ProviderState, StandardSlidingPanelContent } from '../types';
export interface ConnectedAccountPaymentMethodProps {} export interface ConnectedAccountPaymentMethodProps {}
interface ConnectedState { interface ConnectedState {
network: Network;
providerState: ProviderState;
}
interface ConnectedDispatch {
onInstallWalletClick: () => void;
unlockWalletAndDispatchToStore: (providerState: ProviderState) => void;
}
interface ConnectedProps {
onInstallWalletClick: () => void;
onUnlockWalletClick: () => void;
account: Account; account: Account;
network: Network; network: Network;
} }
interface ConnectedDispatch { type FinalProps = ConnectedProps & ConnectedAccountPaymentMethodProps;
openStandardSlidingPanel: (content: StandardSlidingPanelContent) => void;
}
const mapStateToProps = (state: State, _ownProps: ConnectedAccountPaymentMethodProps): ConnectedState => ({ const mapStateToProps = (state: State, _ownProps: ConnectedAccountPaymentMethodProps): ConnectedState => ({
account: state.providerState.account,
network: state.network, network: state.network,
providerState: state.providerState,
}); });
const mapDispatchToProps = ( const mapDispatchToProps = (
dispatch: Dispatch<Action>, dispatch: Dispatch<Action>,
ownProps: ConnectedAccountPaymentMethodProps, ownProps: ConnectedAccountPaymentMethodProps,
): ConnectedDispatch => ({ ): ConnectedDispatch => ({
openStandardSlidingPanel: (content: StandardSlidingPanelContent) => onInstallWalletClick: () => dispatch(actions.openStandardSlidingPanel(StandardSlidingPanelContent.InstallWallet)),
dispatch(actions.openStandardSlidingPanel(content)), unlockWalletAndDispatchToStore: async (providerState: ProviderState) =>
asyncData.fetchAccountInfoAndDispatchToStore(providerState, dispatch, true),
});
const mergeProps = (
connectedState: ConnectedState,
connectedDispatch: ConnectedDispatch,
ownProps: ConnectedAccountPaymentMethodProps,
): FinalProps => ({
...ownProps,
network: connectedState.network,
account: connectedState.providerState.account,
onInstallWalletClick: connectedDispatch.onInstallWalletClick,
onUnlockWalletClick: () => {
connectedDispatch.unlockWalletAndDispatchToStore(connectedState.providerState);
},
}); });
export const ConnectedAccountPaymentMethod: React.ComponentClass<ConnectedAccountPaymentMethodProps> = connect( export const ConnectedAccountPaymentMethod: React.ComponentClass<ConnectedAccountPaymentMethodProps> = connect(
mapStateToProps, mapStateToProps,
mapDispatchToProps, mapDispatchToProps,
mergeProps,
)(PaymentMethod); )(PaymentMethod);

View File

@ -1,71 +1,75 @@
import { AssetProxyId } from '@0x/types'; import { AssetProxyId } from '@0x/types';
import * as _ from 'lodash'; import * as _ from 'lodash';
import { Dispatch } from 'redux';
import { BIG_NUMBER_ZERO } from '../constants'; import { BIG_NUMBER_ZERO } from '../constants';
import { AccountState, ERC20Asset, OrderProcessState } from '../types'; import { AccountState, ERC20Asset, OrderProcessState, ProviderState } from '../types';
import { assetUtils } from '../util/asset'; import { assetUtils } from '../util/asset';
import { buyQuoteUpdater } from '../util/buy_quote_updater'; import { buyQuoteUpdater } from '../util/buy_quote_updater';
import { coinbaseApi } from '../util/coinbase_api'; import { coinbaseApi } from '../util/coinbase_api';
import { errorFlasher } from '../util/error_flasher'; import { errorFlasher } from '../util/error_flasher';
import { actions } from './actions'; import { actions } from './actions';
import { Store } from './store'; import { State } from './reducer';
export const asyncData = { export const asyncData = {
fetchEthPriceAndDispatchToStore: async (store: Store) => { fetchEthPriceAndDispatchToStore: async (dispatch: Dispatch) => {
try { try {
const ethUsdPrice = await coinbaseApi.getEthUsdPrice(); const ethUsdPrice = await coinbaseApi.getEthUsdPrice();
store.dispatch(actions.updateEthUsdPrice(ethUsdPrice)); dispatch(actions.updateEthUsdPrice(ethUsdPrice));
} catch (e) { } catch (e) {
const errorMessage = 'Error fetching ETH/USD price'; const errorMessage = 'Error fetching ETH/USD price';
errorFlasher.flashNewErrorMessage(store.dispatch, errorMessage); errorFlasher.flashNewErrorMessage(dispatch, errorMessage);
store.dispatch(actions.updateEthUsdPrice(BIG_NUMBER_ZERO)); dispatch(actions.updateEthUsdPrice(BIG_NUMBER_ZERO));
} }
}, },
fetchAvailableAssetDatasAndDispatchToStore: async (store: Store) => { fetchAvailableAssetDatasAndDispatchToStore: async (state: State, dispatch: Dispatch) => {
const { providerState, assetMetaDataMap, network } = store.getState(); const { providerState, assetMetaDataMap, network } = state;
const assetBuyer = providerState.assetBuyer; const assetBuyer = providerState.assetBuyer;
try { try {
const assetDatas = await assetBuyer.getAvailableAssetDatasAsync(); const assetDatas = await assetBuyer.getAvailableAssetDatasAsync();
const assets = assetUtils.createAssetsFromAssetDatas(assetDatas, assetMetaDataMap, network); const assets = assetUtils.createAssetsFromAssetDatas(assetDatas, assetMetaDataMap, network);
store.dispatch(actions.setAvailableAssets(assets)); dispatch(actions.setAvailableAssets(assets));
} catch (e) { } catch (e) {
const errorMessage = 'Could not find any assets'; const errorMessage = 'Could not find any assets';
errorFlasher.flashNewErrorMessage(store.dispatch, errorMessage); errorFlasher.flashNewErrorMessage(dispatch, errorMessage);
// On error, just specify that none are available // On error, just specify that none are available
store.dispatch(actions.setAvailableAssets([])); dispatch(actions.setAvailableAssets([]));
} }
}, },
fetchAccountInfoAndDispatchToStore: async (options: { store: Store; shouldSetToLoading: boolean }) => { fetchAccountInfoAndDispatchToStore: async (
const { store, shouldSetToLoading } = options; providerState: ProviderState,
const { providerState } = store.getState(); dispatch: Dispatch,
shouldAttemptUnlock: boolean = false,
shouldSetToLoading: boolean = false,
) => {
const web3Wrapper = providerState.web3Wrapper; const web3Wrapper = providerState.web3Wrapper;
const provider = providerState.provider; const provider = providerState.provider;
if (shouldSetToLoading && providerState.account.state !== AccountState.Loading) { if (shouldSetToLoading && providerState.account.state !== AccountState.Loading) {
store.dispatch(actions.setAccountStateLoading()); dispatch(actions.setAccountStateLoading());
} }
let availableAddresses: string[]; let availableAddresses: string[];
try { try {
// TODO(bmillman): Add support at the web3Wrapper level for calling `eth_requestAccounts` instead of calling enable here // TODO(bmillman): Add support at the web3Wrapper level for calling `eth_requestAccounts` instead of calling enable here
const isPrivacyModeEnabled = !_.isUndefined((provider as any).enable); const isPrivacyModeEnabled = !_.isUndefined((provider as any).enable);
availableAddresses = isPrivacyModeEnabled availableAddresses =
? await (provider as any).enable() isPrivacyModeEnabled && shouldAttemptUnlock
: await web3Wrapper.getAvailableAddressesAsync(); ? await (provider as any).enable()
: await web3Wrapper.getAvailableAddressesAsync();
} catch (e) { } catch (e) {
store.dispatch(actions.setAccountStateLocked()); dispatch(actions.setAccountStateLocked());
return; return;
} }
if (!_.isEmpty(availableAddresses)) { if (!_.isEmpty(availableAddresses)) {
const activeAddress = availableAddresses[0]; const activeAddress = availableAddresses[0];
store.dispatch(actions.setAccountStateReady(activeAddress)); dispatch(actions.setAccountStateReady(activeAddress));
// tslint:disable-next-line:no-floating-promises // tslint:disable-next-line:no-floating-promises
asyncData.fetchAccountBalanceAndDispatchToStore(store); asyncData.fetchAccountBalanceAndDispatchToStore(providerState, dispatch);
} else { } else {
store.dispatch(actions.setAccountStateLocked()); dispatch(actions.setAccountStateLocked());
} }
}, },
fetchAccountBalanceAndDispatchToStore: async (store: Store) => { fetchAccountBalanceAndDispatchToStore: async (providerState: ProviderState, dispatch: Dispatch) => {
const { providerState } = store.getState();
const web3Wrapper = providerState.web3Wrapper; const web3Wrapper = providerState.web3Wrapper;
const account = providerState.account; const account = providerState.account;
if (account.state !== AccountState.Ready) { if (account.state !== AccountState.Ready) {
@ -74,15 +78,18 @@ export const asyncData = {
try { try {
const address = account.address; const address = account.address;
const ethBalanceInWei = await web3Wrapper.getBalanceInWeiAsync(address); const ethBalanceInWei = await web3Wrapper.getBalanceInWeiAsync(address);
store.dispatch(actions.updateAccountEthBalance({ address, ethBalanceInWei })); dispatch(actions.updateAccountEthBalance({ address, ethBalanceInWei }));
} catch (e) { } catch (e) {
// leave balance as is // leave balance as is
return; return;
} }
}, },
fetchCurrentBuyQuoteAndDispatchToStore: async (options: { store: Store; shouldSetPending: boolean }) => { fetchCurrentBuyQuoteAndDispatchToStore: async (
const { store, shouldSetPending } = options; state: State,
const { buyOrderState, providerState, selectedAsset, selectedAssetAmount, affiliateInfo } = store.getState(); dispatch: Dispatch,
shouldSetPending: boolean = false,
) => {
const { buyOrderState, providerState, selectedAsset, selectedAssetAmount, affiliateInfo } = state;
const assetBuyer = providerState.assetBuyer; const assetBuyer = providerState.assetBuyer;
if ( if (
!_.isUndefined(selectedAssetAmount) && !_.isUndefined(selectedAssetAmount) &&
@ -92,7 +99,7 @@ export const asyncData = {
) { ) {
await buyQuoteUpdater.updateBuyQuoteAsync( await buyQuoteUpdater.updateBuyQuoteAsync(
assetBuyer, assetBuyer,
store.dispatch, dispatch,
selectedAsset as ERC20Asset, selectedAsset as ERC20Asset,
selectedAssetAmount, selectedAssetAmount,
shouldSetPending, shouldSetPending,

View File

@ -10,13 +10,13 @@ export interface HeartbeatFactoryOptions {
export const generateAccountHeartbeater = (options: HeartbeatFactoryOptions): Heartbeater => { export const generateAccountHeartbeater = (options: HeartbeatFactoryOptions): Heartbeater => {
const { store, shouldPerformImmediatelyOnStart } = options; const { store, shouldPerformImmediatelyOnStart } = options;
return new Heartbeater(async () => { return new Heartbeater(async () => {
await asyncData.fetchAccountInfoAndDispatchToStore({ store, shouldSetToLoading: false }); await asyncData.fetchAccountInfoAndDispatchToStore(store.getState().providerState, store.dispatch, false);
}, shouldPerformImmediatelyOnStart); }, shouldPerformImmediatelyOnStart);
}; };
export const generateBuyQuoteHeartbeater = (options: HeartbeatFactoryOptions): Heartbeater => { export const generateBuyQuoteHeartbeater = (options: HeartbeatFactoryOptions): Heartbeater => {
const { store, shouldPerformImmediatelyOnStart } = options; const { store, shouldPerformImmediatelyOnStart } = options;
return new Heartbeater(async () => { return new Heartbeater(async () => {
await asyncData.fetchCurrentBuyQuoteAndDispatchToStore({ store, shouldSetPending: false }); await asyncData.fetchCurrentBuyQuoteAndDispatchToStore(store.getState(), store.dispatch, false);
}, shouldPerformImmediatelyOnStart); }, shouldPerformImmediatelyOnStart);
}; };