444 lines
18 KiB
TypeScript
444 lines
18 KiB
TypeScript
import { ZeroEx } from '0x.js';
|
|
import {
|
|
colors,
|
|
constants as sharedConstants,
|
|
EtherscanLinkSuffixes,
|
|
Styles,
|
|
utils as sharedUtils,
|
|
} from '@0xproject/react-shared';
|
|
import { BigNumber } from '@0xproject/utils';
|
|
import * as _ from 'lodash';
|
|
import FlatButton from 'material-ui/FlatButton';
|
|
import { List, ListItem } from 'material-ui/List';
|
|
import NavigationArrowDownward from 'material-ui/svg-icons/navigation/arrow-downward';
|
|
import NavigationArrowUpward from 'material-ui/svg-icons/navigation/arrow-upward';
|
|
import Close from 'material-ui/svg-icons/navigation/close';
|
|
import * as React from 'react';
|
|
import ReactTooltip = require('react-tooltip');
|
|
import firstBy = require('thenby');
|
|
|
|
import { Blockchain } from 'ts/blockchain';
|
|
import { AllowanceToggle } from 'ts/components/inputs/allowance_toggle';
|
|
import { Identicon } from 'ts/components/ui/identicon';
|
|
import { TokenIcon } from 'ts/components/ui/token_icon';
|
|
import { WrapEtherItem } from 'ts/components/wallet/wrap_ether_item';
|
|
import { Dispatcher } from 'ts/redux/dispatcher';
|
|
import { BalanceErrs, BlockchainErrs, Side, Token, TokenByAddress, TokenState, TokenStateByAddress } from 'ts/types';
|
|
import { constants } from 'ts/utils/constants';
|
|
import { utils } from 'ts/utils/utils';
|
|
import { styles as walletItemStyles } from 'ts/utils/wallet_item_styles';
|
|
|
|
export interface WalletProps {
|
|
userAddress: string;
|
|
networkId: number;
|
|
blockchain: Blockchain;
|
|
blockchainIsLoaded: boolean;
|
|
blockchainErr: BlockchainErrs;
|
|
dispatcher: Dispatcher;
|
|
tokenByAddress: TokenByAddress;
|
|
trackedTokens: Token[];
|
|
userEtherBalanceInWei: BigNumber;
|
|
lastForceTokenStateRefetch: number;
|
|
}
|
|
|
|
interface WalletState {
|
|
trackedTokenStateByAddress: TokenStateByAddress;
|
|
wrappedEtherDirection?: Side;
|
|
}
|
|
|
|
interface AllowanceToggleConfig {
|
|
token: Token;
|
|
tokenState: TokenState;
|
|
}
|
|
|
|
interface AccessoryItemConfig {
|
|
wrappedEtherDirection?: Side;
|
|
allowanceToggleConfig?: AllowanceToggleConfig;
|
|
}
|
|
|
|
const styles: Styles = {
|
|
wallet: {
|
|
width: 346,
|
|
backgroundColor: colors.white,
|
|
borderBottomRightRadius: 10,
|
|
borderBottomLeftRadius: 10,
|
|
borderTopRightRadius: 10,
|
|
borderTopLeftRadius: 10,
|
|
boxShadow: `0px 4px 6px ${colors.walletBoxShadow}`,
|
|
overflow: 'hidden',
|
|
},
|
|
list: {
|
|
padding: 0,
|
|
},
|
|
tokenItemInnerDiv: {
|
|
paddingLeft: 60,
|
|
},
|
|
headerItemInnerDiv: {
|
|
paddingLeft: 65,
|
|
},
|
|
footerItemInnerDiv: {
|
|
paddingLeft: 24,
|
|
},
|
|
borderedItem: {
|
|
borderBottomColor: colors.walletBorder,
|
|
borderBottomStyle: 'solid',
|
|
borderWidth: 1,
|
|
},
|
|
tokenItem: {
|
|
backgroundColor: colors.walletDefaultItemBackground,
|
|
},
|
|
wrappedEtherOpenButtonLabel: {
|
|
fontSize: 10,
|
|
},
|
|
amountLabel: {
|
|
fontWeight: 'bold',
|
|
color: colors.black,
|
|
},
|
|
paddedItem: {
|
|
paddingTop: 8,
|
|
paddingBottom: 8,
|
|
},
|
|
accessoryItemsContainer: { width: 150, right: 8 },
|
|
};
|
|
|
|
const ETHER_ICON_PATH = '/images/ether.png';
|
|
const ETHER_TOKEN_SYMBOL = 'WETH';
|
|
const ZRX_TOKEN_SYMBOL = 'ZRX';
|
|
const ETHER_SYMBOL = 'ETH';
|
|
const ICON_DIMENSION = 24;
|
|
const TOKEN_AMOUNT_DISPLAY_PRECISION = 3;
|
|
|
|
export class Wallet extends React.Component<WalletProps, WalletState> {
|
|
private _isUnmounted: boolean;
|
|
constructor(props: WalletProps) {
|
|
super(props);
|
|
this._isUnmounted = false;
|
|
const initialTrackedTokenStateByAddress = this._getInitialTrackedTokenStateByAddress(props.trackedTokens);
|
|
this.state = {
|
|
trackedTokenStateByAddress: initialTrackedTokenStateByAddress,
|
|
wrappedEtherDirection: undefined,
|
|
};
|
|
}
|
|
public componentWillMount() {
|
|
const trackedTokenAddresses = _.keys(this.state.trackedTokenStateByAddress);
|
|
// tslint:disable-next-line:no-floating-promises
|
|
this._fetchBalancesAndAllowancesAsync(trackedTokenAddresses);
|
|
}
|
|
public componentWillUnmount() {
|
|
this._isUnmounted = true;
|
|
}
|
|
public componentWillReceiveProps(nextProps: WalletProps) {
|
|
if (
|
|
nextProps.userAddress !== this.props.userAddress ||
|
|
nextProps.networkId !== this.props.networkId ||
|
|
nextProps.lastForceTokenStateRefetch !== this.props.lastForceTokenStateRefetch
|
|
) {
|
|
const trackedTokenAddresses = _.keys(this.state.trackedTokenStateByAddress);
|
|
// tslint:disable-next-line:no-floating-promises
|
|
this._fetchBalancesAndAllowancesAsync(trackedTokenAddresses);
|
|
}
|
|
if (!_.isEqual(nextProps.trackedTokens, this.props.trackedTokens)) {
|
|
const newTokens = _.difference(nextProps.trackedTokens, this.props.trackedTokens);
|
|
const newTokenAddresses = _.map(newTokens, token => token.address);
|
|
// Add placeholder entry for this token to the state, since fetching the
|
|
// balance/allowance is asynchronous
|
|
const trackedTokenStateByAddress = this.state.trackedTokenStateByAddress;
|
|
_.each(newTokenAddresses, (tokenAddress: string) => {
|
|
trackedTokenStateByAddress[tokenAddress] = {
|
|
balance: new BigNumber(0),
|
|
allowance: new BigNumber(0),
|
|
isLoaded: false,
|
|
};
|
|
});
|
|
this.setState({
|
|
trackedTokenStateByAddress,
|
|
});
|
|
// Fetch the actual balance/allowance.
|
|
// tslint:disable-next-line:no-floating-promises
|
|
this._fetchBalancesAndAllowancesAsync(newTokenAddresses);
|
|
}
|
|
}
|
|
public render() {
|
|
const isReadyToRender = this.props.blockchainIsLoaded && this.props.blockchainErr === BlockchainErrs.NoError;
|
|
return <div style={styles.wallet}>{isReadyToRender && this._renderRows()}</div>;
|
|
}
|
|
private _renderRows() {
|
|
return (
|
|
<List style={styles.list}>
|
|
{_.concat(
|
|
this._renderHeaderRows(),
|
|
this._renderEthRows(),
|
|
this._renderTokenRows(),
|
|
this._renderFooterRows(),
|
|
)}
|
|
</List>
|
|
);
|
|
}
|
|
private _renderHeaderRows() {
|
|
const userAddress = this.props.userAddress;
|
|
const primaryText = utils.getAddressBeginAndEnd(userAddress);
|
|
return (
|
|
<ListItem
|
|
primaryText={primaryText}
|
|
leftIcon={<Identicon address={userAddress} diameter={ICON_DIMENSION} />}
|
|
style={{ ...styles.paddedItem, ...styles.borderedItem }}
|
|
innerDivStyle={styles.headerItemInnerDiv}
|
|
/>
|
|
);
|
|
}
|
|
private _renderFooterRows() {
|
|
const primaryText = '+ other tokens';
|
|
return <ListItem primaryText={primaryText} innerDivStyle={styles.footerItemInnerDiv} />;
|
|
}
|
|
private _renderEthRows() {
|
|
const primaryText = this._renderAmount(
|
|
this.props.userEtherBalanceInWei,
|
|
constants.DECIMAL_PLACES_ETH,
|
|
ETHER_SYMBOL,
|
|
);
|
|
const accessoryItemConfig = {
|
|
wrappedEtherDirection: Side.Deposit,
|
|
};
|
|
const isInWrappedEtherState =
|
|
!_.isUndefined(this.state.wrappedEtherDirection) &&
|
|
this.state.wrappedEtherDirection === accessoryItemConfig.wrappedEtherDirection;
|
|
const style = isInWrappedEtherState
|
|
? { ...walletItemStyles.focusedItem, ...styles.paddedItem }
|
|
: { ...styles.tokenItem, ...styles.borderedItem, ...styles.paddedItem };
|
|
const etherToken = this._getEthToken();
|
|
return (
|
|
<div>
|
|
<ListItem
|
|
primaryText={primaryText}
|
|
leftIcon={<img style={{ width: ICON_DIMENSION, height: ICON_DIMENSION }} src={ETHER_ICON_PATH} />}
|
|
rightAvatar={this._renderAccessoryItems(accessoryItemConfig)}
|
|
disableTouchRipple={true}
|
|
style={style}
|
|
innerDivStyle={styles.tokenItemInnerDiv}
|
|
/>
|
|
{isInWrappedEtherState && (
|
|
<WrapEtherItem
|
|
userAddress={this.props.userAddress}
|
|
networkId={this.props.networkId}
|
|
blockchain={this.props.blockchain}
|
|
dispatcher={this.props.dispatcher}
|
|
userEtherBalanceInWei={this.props.userEtherBalanceInWei}
|
|
direction={accessoryItemConfig.wrappedEtherDirection}
|
|
etherToken={etherToken}
|
|
lastForceTokenStateRefetch={this.props.lastForceTokenStateRefetch}
|
|
onConversionSuccessful={this._closeWrappedEtherActionRow.bind(this)}
|
|
refetchEthTokenStateAsync={this._refetchTokenStateAsync.bind(this, etherToken.address)}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
private _renderTokenRows() {
|
|
const trackedTokens = this.props.trackedTokens;
|
|
const trackedTokensStartingWithEtherToken = trackedTokens.sort(
|
|
firstBy((t: Token) => t.symbol !== ETHER_TOKEN_SYMBOL)
|
|
.thenBy((t: Token) => t.symbol !== ZRX_TOKEN_SYMBOL)
|
|
.thenBy('address'),
|
|
);
|
|
return _.map(trackedTokensStartingWithEtherToken, this._renderTokenRow.bind(this));
|
|
}
|
|
private _renderTokenRow(token: Token) {
|
|
const tokenState = this.state.trackedTokenStateByAddress[token.address];
|
|
const tokenLink = sharedUtils.getEtherScanLinkIfExists(
|
|
token.address,
|
|
this.props.networkId,
|
|
EtherscanLinkSuffixes.Address,
|
|
);
|
|
const amount = this._renderAmount(tokenState.balance, token.decimals, token.symbol);
|
|
const wrappedEtherDirection = token.symbol === ETHER_TOKEN_SYMBOL ? Side.Receive : undefined;
|
|
const accessoryItemConfig: AccessoryItemConfig = {
|
|
wrappedEtherDirection,
|
|
allowanceToggleConfig: {
|
|
token,
|
|
tokenState,
|
|
},
|
|
};
|
|
const shouldShowWrapEtherItem =
|
|
!_.isUndefined(this.state.wrappedEtherDirection) &&
|
|
this.state.wrappedEtherDirection === accessoryItemConfig.wrappedEtherDirection;
|
|
const style = shouldShowWrapEtherItem
|
|
? { ...walletItemStyles.focusedItem, ...styles.paddedItem }
|
|
: { ...styles.tokenItem, ...styles.borderedItem, ...styles.paddedItem };
|
|
const etherToken = this._getEthToken();
|
|
return (
|
|
<div>
|
|
<ListItem
|
|
primaryText={amount}
|
|
leftIcon={this._renderTokenIcon(token, tokenLink)}
|
|
rightAvatar={this._renderAccessoryItems(accessoryItemConfig)}
|
|
disableTouchRipple={true}
|
|
style={style}
|
|
innerDivStyle={styles.tokenItemInnerDiv}
|
|
/>
|
|
{shouldShowWrapEtherItem && (
|
|
<WrapEtherItem
|
|
userAddress={this.props.userAddress}
|
|
networkId={this.props.networkId}
|
|
blockchain={this.props.blockchain}
|
|
dispatcher={this.props.dispatcher}
|
|
userEtherBalanceInWei={this.props.userEtherBalanceInWei}
|
|
direction={accessoryItemConfig.wrappedEtherDirection}
|
|
etherToken={etherToken}
|
|
lastForceTokenStateRefetch={this.props.lastForceTokenStateRefetch}
|
|
onConversionSuccessful={this._closeWrappedEtherActionRow.bind(this)}
|
|
refetchEthTokenStateAsync={this._refetchTokenStateAsync.bind(this, etherToken.address)}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
private _renderAccessoryItems(config: AccessoryItemConfig) {
|
|
const shouldShowWrappedEtherAction = !_.isUndefined(config.wrappedEtherDirection);
|
|
const shouldShowToggle = !_.isUndefined(config.allowanceToggleConfig);
|
|
return (
|
|
<div style={styles.accessoryItemsContainer}>
|
|
<div className="flex">
|
|
<div className="flex-auto">
|
|
{shouldShowWrappedEtherAction && this._renderWrappedEtherButton(config.wrappedEtherDirection)}
|
|
</div>
|
|
<div className="flex-last py1">
|
|
{shouldShowToggle && this._renderAllowanceToggle(config.allowanceToggleConfig)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
private _renderAllowanceToggle(config: AllowanceToggleConfig) {
|
|
return (
|
|
<AllowanceToggle
|
|
networkId={this.props.networkId}
|
|
blockchain={this.props.blockchain}
|
|
dispatcher={this.props.dispatcher}
|
|
token={config.token}
|
|
tokenState={config.tokenState}
|
|
onErrorOccurred={_.noop} // TODO: Error handling
|
|
userAddress={this.props.userAddress}
|
|
isDisabled={!config.tokenState.isLoaded}
|
|
refetchTokenStateAsync={this._refetchTokenStateAsync.bind(this, config.token.address)}
|
|
/>
|
|
);
|
|
}
|
|
private _renderAmount(amount: BigNumber, decimals: number, symbol: string) {
|
|
const unitAmount = ZeroEx.toUnitAmount(amount, decimals);
|
|
const formattedAmount = unitAmount.toPrecision(TOKEN_AMOUNT_DISPLAY_PRECISION);
|
|
const result = `${formattedAmount} ${symbol}`;
|
|
return <div style={styles.amountLabel}>{result}</div>;
|
|
}
|
|
private _renderTokenIcon(token: Token, tokenLink?: string) {
|
|
const tooltipId = `tooltip-${token.address}`;
|
|
const tokenIcon = <TokenIcon token={token} diameter={ICON_DIMENSION} />;
|
|
if (_.isUndefined(tokenLink)) {
|
|
return tokenIcon;
|
|
} else {
|
|
return (
|
|
<a href={tokenLink} target="_blank" style={{ textDecoration: 'none' }}>
|
|
{tokenIcon}
|
|
</a>
|
|
);
|
|
}
|
|
}
|
|
private _renderWrappedEtherButton(wrappedEtherDirection: Side) {
|
|
const isWrappedEtherDirectionOpen = this.state.wrappedEtherDirection === wrappedEtherDirection;
|
|
let buttonLabel;
|
|
let buttonIcon;
|
|
if (isWrappedEtherDirectionOpen) {
|
|
buttonLabel = 'cancel';
|
|
buttonIcon = <Close />;
|
|
} else {
|
|
switch (wrappedEtherDirection) {
|
|
case Side.Deposit:
|
|
buttonLabel = 'wrap';
|
|
buttonIcon = <NavigationArrowDownward />;
|
|
break;
|
|
case Side.Receive:
|
|
buttonLabel = 'unwrap';
|
|
buttonIcon = <NavigationArrowUpward />;
|
|
break;
|
|
default:
|
|
throw utils.spawnSwitchErr('wrappedEtherDirection', wrappedEtherDirection);
|
|
}
|
|
}
|
|
const onClick = isWrappedEtherDirectionOpen
|
|
? this._closeWrappedEtherActionRow.bind(this)
|
|
: this._openWrappedEtherActionRow.bind(this, wrappedEtherDirection);
|
|
return (
|
|
<FlatButton
|
|
label={buttonLabel}
|
|
labelPosition="after"
|
|
primary={true}
|
|
icon={buttonIcon}
|
|
labelStyle={styles.wrappedEtherOpenButtonLabel}
|
|
onClick={onClick}
|
|
/>
|
|
);
|
|
}
|
|
private _getInitialTrackedTokenStateByAddress(trackedTokens: Token[]) {
|
|
const trackedTokenStateByAddress: TokenStateByAddress = {};
|
|
_.each(trackedTokens, token => {
|
|
trackedTokenStateByAddress[token.address] = {
|
|
balance: new BigNumber(0),
|
|
allowance: new BigNumber(0),
|
|
isLoaded: false,
|
|
};
|
|
});
|
|
return trackedTokenStateByAddress;
|
|
}
|
|
private async _fetchBalancesAndAllowancesAsync(tokenAddresses: string[]) {
|
|
const trackedTokenStateByAddress = this.state.trackedTokenStateByAddress;
|
|
const userAddressIfExists = _.isEmpty(this.props.userAddress) ? undefined : this.props.userAddress;
|
|
for (const tokenAddress of tokenAddresses) {
|
|
const [balance, allowance] = await this.props.blockchain.getTokenBalanceAndAllowanceAsync(
|
|
userAddressIfExists,
|
|
tokenAddress,
|
|
);
|
|
trackedTokenStateByAddress[tokenAddress] = {
|
|
balance,
|
|
allowance,
|
|
isLoaded: true,
|
|
};
|
|
}
|
|
if (!this._isUnmounted) {
|
|
this.setState({
|
|
trackedTokenStateByAddress,
|
|
});
|
|
}
|
|
}
|
|
private async _refetchTokenStateAsync(tokenAddress: string) {
|
|
const userAddressIfExists = _.isEmpty(this.props.userAddress) ? undefined : this.props.userAddress;
|
|
const [balance, allowance] = await this.props.blockchain.getTokenBalanceAndAllowanceAsync(
|
|
userAddressIfExists,
|
|
tokenAddress,
|
|
);
|
|
this.setState({
|
|
trackedTokenStateByAddress: {
|
|
...this.state.trackedTokenStateByAddress,
|
|
[tokenAddress]: {
|
|
balance,
|
|
allowance,
|
|
isLoaded: true,
|
|
},
|
|
},
|
|
});
|
|
}
|
|
private _openWrappedEtherActionRow(wrappedEtherDirection: Side) {
|
|
this.setState({
|
|
wrappedEtherDirection,
|
|
});
|
|
}
|
|
private _closeWrappedEtherActionRow() {
|
|
this.setState({
|
|
wrappedEtherDirection: undefined,
|
|
});
|
|
}
|
|
private _getEthToken() {
|
|
const tokens = _.values(this.props.tokenByAddress);
|
|
const etherToken = _.find(tokens, { symbol: ETHER_TOKEN_SYMBOL });
|
|
return etherToken;
|
|
}
|
|
}
|