mirror of
https://github.com/status-im/MyCrypto.git
synced 2025-01-22 08:58:55 +00:00
a9f78011cb
Fixes #226 Fixes #383 Added a simple check to ensure that swap rates exist so we don't get a NaN error, which would be confusing to users. Instead, the form is hidden and a spinner is shown until the bityRates are ready for the user. * adds spinner while fetching * added error on top of input * removed classnames prop and added cn * added simple err mssge * css fixes and disabled button * better err mssge generation and fixed swap details * minor typo * added redux notification on unreachable error * minheight for err message and swap update on redux change * fixed formatting and removed className prop
348 lines
10 KiB
TypeScript
348 lines
10 KiB
TypeScript
import { TShowNotification } from 'actions/notifications';
|
|
import {
|
|
TChangeStepSwap,
|
|
TDestinationAmountSwap,
|
|
TDestinationKindSwap,
|
|
TOriginAmountSwap,
|
|
TOriginKindSwap
|
|
} from 'actions/swap';
|
|
import SimpleButton from 'components/ui/SimpleButton';
|
|
import bityConfig, { generateKindMax, generateKindMin } from 'config/bity';
|
|
import React, { Component } from 'react';
|
|
import translate from 'translations';
|
|
import { combineAndUpper, toFixedIfLarger } from 'utils/formatters';
|
|
import './CurrencySwap.scss';
|
|
import { Dropdown } from 'components/ui';
|
|
import Spinner from 'components/ui/Spinner';
|
|
|
|
export interface StateProps {
|
|
bityRates: any;
|
|
originAmount: number | null;
|
|
destinationAmount: number | null;
|
|
originKind: string;
|
|
destinationKind: string;
|
|
destinationKindOptions: string[];
|
|
originKindOptions: string[];
|
|
}
|
|
|
|
export interface ActionProps {
|
|
showNotification: TShowNotification;
|
|
changeStepSwap: TChangeStepSwap;
|
|
originKindSwap: TOriginKindSwap;
|
|
destinationKindSwap: TDestinationKindSwap;
|
|
originAmountSwap: TOriginAmountSwap;
|
|
destinationAmountSwap: TDestinationAmountSwap;
|
|
}
|
|
|
|
interface State {
|
|
disabled: boolean;
|
|
showedMinMaxError: boolean;
|
|
originErr: string;
|
|
destinationErr: string;
|
|
}
|
|
|
|
export default class CurrencySwap extends Component<
|
|
StateProps & ActionProps,
|
|
State
|
|
> {
|
|
public state = {
|
|
disabled: true,
|
|
showedMinMaxError: false,
|
|
originErr: '',
|
|
destinationErr: ''
|
|
};
|
|
|
|
public componentWillReceiveProps(newProps) {
|
|
const {
|
|
originAmount,
|
|
originKind,
|
|
destinationKind,
|
|
destinationAmount
|
|
} = newProps;
|
|
if (
|
|
originKind !== this.props.originKind ||
|
|
destinationKind !== this.props.destinationKind
|
|
) {
|
|
this.setDisabled(
|
|
originAmount,
|
|
originKind,
|
|
destinationKind,
|
|
destinationAmount
|
|
);
|
|
}
|
|
}
|
|
|
|
public isMinMaxValid = (amount, kind) => {
|
|
let bityMin;
|
|
let bityMax;
|
|
if (kind !== 'BTC') {
|
|
const bityPairRate = this.props.bityRates['BTC' + kind];
|
|
bityMin = generateKindMin(bityPairRate, kind);
|
|
bityMax = generateKindMax(bityPairRate, kind);
|
|
} else {
|
|
bityMin = bityConfig.BTCMin;
|
|
bityMax = bityConfig.BTCMax;
|
|
}
|
|
const higherThanMin = amount >= bityMin;
|
|
const lowerThanMax = amount <= bityMax;
|
|
return higherThanMin && lowerThanMax;
|
|
};
|
|
|
|
public isDisabled = (originAmount, originKind, destinationAmount) => {
|
|
const hasOriginAmountAndDestinationAmount =
|
|
originAmount && destinationAmount;
|
|
const minMaxIsValid = this.isMinMaxValid(originAmount, originKind);
|
|
return !(hasOriginAmountAndDestinationAmount && minMaxIsValid);
|
|
};
|
|
|
|
public setDisabled(
|
|
originAmount,
|
|
originKind,
|
|
destinationKind,
|
|
destinationAmount
|
|
) {
|
|
const disabled = this.isDisabled(
|
|
originAmount,
|
|
originKind,
|
|
destinationAmount
|
|
);
|
|
|
|
if (disabled && originAmount) {
|
|
const { bityRates } = this.props;
|
|
const ETHMin = generateKindMin(bityRates.BTCETH, 'ETH');
|
|
const ETHMax = generateKindMax(bityRates.BTCETH, 'ETH');
|
|
const REPMin = generateKindMin(bityRates.BTCREP, 'REP');
|
|
|
|
const getRates = kind => {
|
|
let minAmount;
|
|
let maxAmount;
|
|
switch (kind) {
|
|
case 'BTC':
|
|
minAmount = toFixedIfLarger(bityConfig.BTCMin, 3);
|
|
maxAmount = toFixedIfLarger(bityConfig.BTCMax, 3);
|
|
break;
|
|
case 'ETH':
|
|
minAmount = toFixedIfLarger(ETHMin, 3);
|
|
maxAmount = toFixedIfLarger(ETHMax, 3);
|
|
break;
|
|
case 'REP':
|
|
minAmount = toFixedIfLarger(REPMin, 3);
|
|
break;
|
|
default:
|
|
if (this.state.showedMinMaxError) {
|
|
this.setState(
|
|
{
|
|
showedMinMaxError: true
|
|
},
|
|
() => {
|
|
this.props.showNotification(
|
|
'danger',
|
|
"Couldn't get match currency kind. Something went terribly wrong",
|
|
10000
|
|
);
|
|
}
|
|
);
|
|
}
|
|
}
|
|
return { minAmount, maxAmount };
|
|
};
|
|
|
|
const createErrString = (kind, amount, rate) => {
|
|
let errString;
|
|
if (amount > rate.maxAmount) {
|
|
errString = `Maximum ${kind} is ${rate.maxAmount} ${kind}`;
|
|
} else {
|
|
errString = `Minimum ${kind} is ${rate.minAmount} ${kind}`;
|
|
}
|
|
|
|
return errString;
|
|
};
|
|
const originRate = getRates(originKind);
|
|
const destinationRate = getRates(destinationKind);
|
|
const originErr = createErrString(originKind, originAmount, originRate);
|
|
const destinationErr = createErrString(
|
|
destinationKind,
|
|
destinationAmount,
|
|
destinationRate
|
|
);
|
|
|
|
this.setState({
|
|
originErr,
|
|
destinationErr,
|
|
disabled: true
|
|
});
|
|
} else {
|
|
this.setState({
|
|
originErr: '',
|
|
destinationErr: '',
|
|
disabled
|
|
});
|
|
}
|
|
}
|
|
|
|
public onClickStartSwap = () => {
|
|
this.props.changeStepSwap(2);
|
|
};
|
|
|
|
public setOriginAndDestinationToNull = () => {
|
|
this.props.originAmountSwap(null);
|
|
this.props.destinationAmountSwap(null);
|
|
this.setDisabled(
|
|
null,
|
|
this.props.originKind,
|
|
this.props.destinationKind,
|
|
null
|
|
);
|
|
};
|
|
|
|
public onChangeOriginAmount = (
|
|
event: React.SyntheticEvent<HTMLInputElement>
|
|
) => {
|
|
const { destinationKind, originKind } = this.props;
|
|
const amount = (event.target as HTMLInputElement).value;
|
|
const originAmountAsNumber = parseFloat(amount);
|
|
if (originAmountAsNumber || originAmountAsNumber === 0) {
|
|
const pairName = combineAndUpper(originKind, destinationKind);
|
|
const bityRate = this.props.bityRates[pairName];
|
|
this.props.originAmountSwap(originAmountAsNumber);
|
|
const destinationAmount = originAmountAsNumber * bityRate;
|
|
this.props.destinationAmountSwap(destinationAmount);
|
|
this.setDisabled(
|
|
originAmountAsNumber,
|
|
originKind,
|
|
destinationKind,
|
|
destinationAmount
|
|
);
|
|
} else {
|
|
this.setOriginAndDestinationToNull();
|
|
}
|
|
};
|
|
|
|
public onChangeDestinationAmount = (
|
|
event: React.SyntheticEvent<HTMLInputElement>
|
|
) => {
|
|
const { destinationKind, originKind } = this.props;
|
|
const amount = (event.target as HTMLInputElement).value;
|
|
const destinationAmountAsNumber = parseFloat(amount);
|
|
if (destinationAmountAsNumber || destinationAmountAsNumber === 0) {
|
|
this.props.destinationAmountSwap(destinationAmountAsNumber);
|
|
const pairNameReversed = combineAndUpper(destinationKind, originKind);
|
|
const bityRate = this.props.bityRates[pairNameReversed];
|
|
const originAmount = destinationAmountAsNumber * bityRate;
|
|
this.props.originAmountSwap(originAmount);
|
|
this.setDisabled(
|
|
originAmount,
|
|
originKind,
|
|
destinationKind,
|
|
destinationAmountAsNumber
|
|
);
|
|
} else {
|
|
this.setOriginAndDestinationToNull();
|
|
}
|
|
};
|
|
|
|
public render() {
|
|
const {
|
|
originAmount,
|
|
destinationAmount,
|
|
originKind,
|
|
destinationKind,
|
|
destinationKindOptions,
|
|
originKindOptions,
|
|
bityRates
|
|
} = this.props;
|
|
|
|
const { originErr, destinationErr } = this.state;
|
|
|
|
const OriginKindDropDown = Dropdown as new () => Dropdown<
|
|
typeof originKind
|
|
>;
|
|
const DestinationKindDropDown = Dropdown as new () => Dropdown<
|
|
typeof destinationKind
|
|
>;
|
|
const pairName = combineAndUpper(originKind, destinationKind);
|
|
const bityLoaded = bityRates[pairName];
|
|
return (
|
|
<article className="CurrencySwap">
|
|
<h1 className="CurrencySwap-title">{translate('SWAP_init_1')}</h1>
|
|
{bityLoaded ? (
|
|
<div className="form-inline CurrencySwap-inner-wrap">
|
|
<div className="CurrencySwap-input-group">
|
|
<span className="CurrencySwap-error-message">{originErr}</span>
|
|
<input
|
|
className={`CurrencySwap-input form-control ${
|
|
String(originAmount) !== '' &&
|
|
this.isMinMaxValid(originAmount, originKind)
|
|
? 'is-valid'
|
|
: 'is-invalid'
|
|
}`}
|
|
type="number"
|
|
placeholder="Amount"
|
|
value={originAmount || originAmount === 0 ? originAmount : ''}
|
|
onChange={this.onChangeOriginAmount}
|
|
/>
|
|
<div className="CurrencySwap-dropdown">
|
|
<OriginKindDropDown
|
|
ariaLabel={`change origin kind. current origin kind ${
|
|
originKind
|
|
}`}
|
|
options={originKindOptions}
|
|
value={originKind}
|
|
onChange={this.props.originKindSwap}
|
|
size="smr"
|
|
color="default"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<h1 className="CurrencySwap-divider">{translate('SWAP_init_2')}</h1>
|
|
<div className="CurrencySwap-input-group">
|
|
<span className="CurrencySwap-error-message">
|
|
{destinationErr}
|
|
</span>
|
|
<input
|
|
className={`CurrencySwap-input form-control ${
|
|
String(destinationAmount) !== '' &&
|
|
this.isMinMaxValid(originAmount, originKind)
|
|
? 'is-valid'
|
|
: 'is-invalid'
|
|
}`}
|
|
type="number"
|
|
placeholder="Amount"
|
|
value={
|
|
destinationAmount || destinationAmount === 0
|
|
? destinationAmount
|
|
: ''
|
|
}
|
|
onChange={this.onChangeDestinationAmount}
|
|
/>
|
|
<div className="CurrencySwap-dropdown">
|
|
<DestinationKindDropDown
|
|
ariaLabel={`change destination kind. current destination kind ${
|
|
destinationKind
|
|
}`}
|
|
options={destinationKindOptions}
|
|
value={destinationKind}
|
|
onChange={this.props.destinationKindSwap}
|
|
size="smr"
|
|
color="default"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
) : (
|
|
<Spinner />
|
|
)}
|
|
|
|
<div className="CurrencySwap-submit">
|
|
<SimpleButton
|
|
onClick={this.onClickStartSwap}
|
|
text={translate('SWAP_init_CTA')}
|
|
disabled={this.state.disabled}
|
|
type="info"
|
|
/>
|
|
</div>
|
|
</article>
|
|
);
|
|
}
|
|
}
|