James Prado 0d5d0cea9a Wallet Loading States & Spinner Update (#334)
* Add disclaimer modal to footer

* Remove duplicate code & unnecessary styles

* Fix formatting noise

* remove un-used css style

* Fix tslint error & add media query for modals

* Nest Media Query

* Replace '???' with Spinner & update spinner

* Add loading states for wallet balances

* Update wallet test

* Remove excess data passed to wallet balance reducer & Fix wallet balance types

* Merge 'develop' into 'loading-indicator'

* Add 'light' prop to Spinner

* Only show spinners when fetching data

* Remove format diff

* Apply naming conventions

* Remove network name when offline
2017-11-14 21:44:55 -06:00

167 lines
4.1 KiB
TypeScript

import { SetBalanceFullfilledAction } from 'actions/wallet/actionTypes';
import {
SetTokenBalancesAction,
SetWalletAction,
WalletAction,
TypeKeys
} from 'actions/wallet';
import { TokenValue } from 'libs/units';
import { BroadcastTransactionStatus } from 'libs/transaction';
import { IWallet, Balance } from 'libs/wallet';
import { getTxFromBroadcastTransactionStatus } from 'selectors/wallet';
export interface State {
inst?: IWallet | null;
// in ETH
balance: Balance | { wei: null };
tokens: {
[key: string]: TokenValue;
};
transactions: BroadcastTransactionStatus[];
}
export const INITIAL_STATE: State = {
inst: null,
balance: { isPending: false, wei: null },
tokens: {},
transactions: []
};
function setWallet(state: State, action: SetWalletAction): State {
return {
...state,
inst: action.payload,
balance: INITIAL_STATE.balance,
tokens: INITIAL_STATE.tokens
};
}
function setBalancePending(state: State): State {
return { ...state, balance: { ...state.balance, isPending: true } };
}
function setBalanceFullfilled(
state: State,
action: SetBalanceFullfilledAction
): State {
return {
...state,
balance: { wei: action.payload, isPending: false }
};
}
function setBalanceRejected(state: State): State {
return { ...state, balance: { ...state.balance, isPending: false } };
}
function setTokenBalances(state: State, action: SetTokenBalancesAction): State {
return { ...state, tokens: { ...state.tokens, ...action.payload } };
}
function handleUpdateTxArray(
transactions: BroadcastTransactionStatus[],
broadcastStatusTx: BroadcastTransactionStatus,
isBroadcasting: boolean,
successfullyBroadcast: boolean
): BroadcastTransactionStatus[] {
return transactions.map(item => {
if (item === broadcastStatusTx) {
return { ...item, isBroadcasting, successfullyBroadcast };
} else {
return { ...item };
}
});
}
function handleTxBroadcastCompleted(
state: State,
signedTx: string,
successfullyBroadcast: boolean
): BroadcastTransactionStatus[] {
const existingTx = getTxFromBroadcastTransactionStatus(
state.transactions,
signedTx
);
if (existingTx) {
const isBroadcasting = false;
return handleUpdateTxArray(
state.transactions,
existingTx,
isBroadcasting,
successfullyBroadcast
);
} else {
return state.transactions;
}
}
function handleBroadcastTxRequested(state: State, signedTx: string) {
const existingTx = getTxFromBroadcastTransactionStatus(
state.transactions,
signedTx
);
const isBroadcasting = true;
const successfullyBroadcast = false;
if (!existingTx) {
return state.transactions.concat([
{
signedTx,
isBroadcasting,
successfullyBroadcast
}
]);
} else {
return handleUpdateTxArray(
state.transactions,
existingTx,
isBroadcasting,
successfullyBroadcast
);
}
}
export function wallet(
state: State = INITIAL_STATE,
action: WalletAction
): State {
switch (action.type) {
case TypeKeys.WALLET_SET:
return setWallet(state, action);
case TypeKeys.WALLET_RESET:
return INITIAL_STATE;
case TypeKeys.WALLET_SET_BALANCE_PENDING:
return setBalancePending(state);
case TypeKeys.WALLET_SET_BALANCE_FULFILLED:
return setBalanceFullfilled(state, action);
case TypeKeys.WALLET_SET_BALANCE_REJECTED:
return setBalanceRejected(state);
case TypeKeys.WALLET_SET_TOKEN_BALANCES:
return setTokenBalances(state, action);
case TypeKeys.WALLET_BROADCAST_TX_REQUESTED:
return {
...state,
transactions: handleBroadcastTxRequested(state, action.payload.signedTx)
};
case TypeKeys.WALLET_BROADCAST_TX_SUCCEEDED:
return {
...state,
transactions: handleTxBroadcastCompleted(
state,
action.payload.signedTx,
true
)
};
case TypeKeys.WALLET_BROADCAST_TX_FAILED:
return {
...state,
transactions: handleTxBroadcastCompleted(
state,
action.payload.signedTx,
false
)
};
default:
return state;
}
}