James Prado 9cac0298a2 Improve accessibility (a11y) (#1267)
* Manage modal focus

* Add isOpen prop to CustomNodeModal

* Remove outline overrides

* Update outline style for inputs

* Fix modal focus management & Cleanup CustomNodeModal

* Add aria-label on modal close button

* Fix modal scroll to top

* Add aria-live property for notifications

* Add aria-busy to Spinner component

* Fix border styles for generatewallet password inputs

* Update token balances inputs

* Remove multiple h1's & Update styles

* Add alt text to all img elements

* Update swap link from bity to shapeshift

* Update aria-labels and alt text

* Only show keystore password input when required

* Revert "Only show keystore password input when required"

This reverts commit 7ec5de52da0982cd3131f365b142f6915638d831.

* address changes requested
2018-03-08 13:28:43 -06:00

194 lines
5.7 KiB
TypeScript

import React, { PureComponent } from 'react';
import { connect } from 'react-redux';
import sample from 'lodash/sample';
import times from 'lodash/times';
import {
NormalizedBityRates,
NormalizedShapeshiftRates,
NormalizedShapeshiftRate
} from 'reducers/swap/types';
import bityLogoWhite from 'assets/images/logo-bity-white.svg';
import shapeshiftLogoWhite from 'assets/images/logo-shapeshift.svg';
import Spinner from 'components/ui/Spinner';
import { bityReferralURL, shapeshiftReferralURL } from 'config';
import translate from 'translations';
import { SHAPESHIFT_WHITELIST } from 'api/shapeshift';
import {
loadShapeshiftRatesRequestedSwap,
TLoadShapeshiftRatesRequestedSwap,
stopLoadShapeshiftRatesSwap,
TStopLoadShapeshiftRatesSwap,
ProviderName
} from 'actions/swap';
import { getOffline } from 'selectors/config';
import Rates from './Rates';
import { AppState } from 'reducers';
import './CurrentRates.scss';
import { Optional } from 'utils/types';
interface StateProps {
isOffline: boolean;
provider: ProviderName;
bityRates: NormalizedBityRates;
shapeshiftRates: NormalizedShapeshiftRates;
}
interface ActionProps {
loadShapeshiftRatesRequestedSwap: TLoadShapeshiftRatesRequestedSwap;
stopLoadShapeshiftRatesSwap: TStopLoadShapeshiftRatesSwap;
}
type Props = StateProps & ActionProps;
class CurrentRates extends PureComponent<Props> {
private shapeShiftRateCache: any = null;
public componentDidMount() {
if (!this.props.isOffline) {
this.loadRates();
}
}
public componentWillReceiveProps(nextProps: Props) {
if (this.props.isOffline && !nextProps.isOffline) {
this.loadRates();
}
}
public componentWillUnmount() {
this.props.stopLoadShapeshiftRatesSwap();
}
public loadRates() {
this.props.loadShapeshiftRatesRequestedSwap();
}
public getRandomSSPairData = (
shapeshiftRates: NormalizedShapeshiftRates
): NormalizedShapeshiftRate => {
const coinOne = sample(SHAPESHIFT_WHITELIST) as string;
const coinTwo = sample(SHAPESHIFT_WHITELIST) as string;
const pair = coinOne + coinTwo;
const pairData = shapeshiftRates.byId[pair];
if (pairData) {
return pairData;
} else {
// if random pairing is unavailable / missing in state
return this.getRandomSSPairData(shapeshiftRates);
}
};
public buildSSPairs = (shapeshiftRates: NormalizedShapeshiftRates, n: number = 4) => {
const pairCollection = times(n, () => this.getRandomSSPairData(shapeshiftRates));
const byId = pairCollection.reduce<{ [id: string]: NormalizedShapeshiftRate }>((acc, cur) => {
acc[cur.id] = cur;
return acc;
}, {});
const allIds = pairCollection.map(SSData => SSData.id);
return {
byId,
allIds
};
};
public isValidRates = (rates: Optional<NormalizedShapeshiftRates>) => {
return rates && rates.allIds && rates.allIds.length > 0;
};
public setupRates = () => {
const { shapeshiftRates, bityRates, provider } = this.props;
let fixedRates;
if (provider === 'bity') {
fixedRates = bityRates;
} else if (provider === 'shapeshift') {
// if ShapeShift rates are valid, filter to 4 random pairs
if (this.isValidRates(shapeshiftRates)) {
if (!this.shapeShiftRateCache) {
fixedRates = this.buildSSPairs(shapeshiftRates);
this.shapeShiftRateCache = fixedRates;
} else {
fixedRates = this.shapeShiftRateCache;
}
} else {
// else, pass along invalid rates. Child component will handle showing spinner until they become valid
fixedRates = shapeshiftRates;
}
}
return fixedRates;
};
public swapEl = (providerURL: string, providerLogo: string, children: any) => {
return (
<article className="SwapRates">
<h3 className="SwapRates-title">{translate('SWAP_rates')}</h3>
<section className="SwapRates-panel row">
{children}
<a
className="SwapRates-panel-logo"
href={providerURL}
target="_blank"
rel="noopener noreferrer"
>
<img src={providerLogo} width={120} height={49} alt="Shapeshift Logo" />
</a>
</section>
</article>
);
};
public render() {
const { provider } = this.props;
const rates = this.setupRates();
const providerLogo = provider === 'shapeshift' ? shapeshiftLogoWhite : bityLogoWhite;
const providerURL = provider === 'shapeshift' ? shapeshiftReferralURL : bityReferralURL;
let children;
if (this.isValidRates(rates)) {
children = <Rates provider={provider} rates={rates} />;
} else {
// TODO - de-dup
children = (
<>
<div className="SwapRates-panel-side col-sm-6">
<div className="SwapRates-panel-rate">
<Spinner size="x1" light={true} />
</div>
<div className="SwapRates-panel-rate">
<Spinner size="x1" light={true} />
</div>
</div>
<div className="SwapRates-panel-side col-sm-6">
<div className="SwapRates-panel-rate">
<Spinner size="x1" light={true} />
</div>
<div className="SwapRates-panel-rate">
<Spinner size="x1" light={true} />
</div>
</div>
</>
);
}
return this.swapEl(providerURL, providerLogo, children);
}
}
function mapStateToProps(state: AppState): StateProps {
return {
isOffline: getOffline(state),
provider: state.swap.provider,
bityRates: state.swap.bityRates,
shapeshiftRates: state.swap.shapeshiftRates
};
}
export default connect(mapStateToProps, {
loadShapeshiftRatesRequestedSwap,
stopLoadShapeshiftRatesSwap
})(CurrentRates);