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

98 lines
2.9 KiB
TypeScript

import React, { Component } from 'react';
import { connect } from 'react-redux';
import { AppState } from 'reducers';
import { resolveDomainRequested, TResolveDomainRequested } from 'actions/ens';
import { isValidENSName } from 'libs/validators';
import './NameInput.scss';
import { Input } from 'components/ui';
interface State {
domainToCheck: string;
isValidDomain: boolean;
isFocused: boolean;
}
interface Props {
domainRequests: AppState['ens']['domainRequests'];
resolveDomainRequested: TResolveDomainRequested;
}
class NameInput extends Component<Props, State> {
public state = {
isFocused: false,
isValidDomain: false,
domainToCheck: ''
};
public render() {
const { domainRequests } = this.props;
const { isValidDomain, domainToCheck, isFocused } = this.state;
const req = domainRequests[domainToCheck];
const isLoading = req && !req.data && !req.error;
return (
<form className="ENSInput" onSubmit={this.onSubmit}>
<div className="input-group-wrapper">
<label className="input-group input-group-inline ENSInput-name">
<Input
value={domainToCheck}
className={`${
!domainToCheck ? '' : isValidDomain ? '' : 'invalid'
} border-rad-right-0`}
type="text"
placeholder="mycrypto"
onChange={this.onChange}
onFocus={this.onFocus}
onBlur={this.onBlur}
disabled={isLoading}
/>
<span className="input-group-addon">.eth</span>
</label>
</div>
{domainToCheck &&
!isValidDomain &&
!isFocused && (
<p className="help-block is-invalid">
Must be at least 7 characters, no special characters
</p>
)}
<button
className="ENSInput-button btn btn-primary btn-block"
disabled={!isValidDomain || isLoading}
>
Check Availability
</button>
</form>
);
}
// add delay to namehash computation / getting the availability
private onChange = (event: React.FormEvent<HTMLButtonElement>) => {
const domainToCheck = event.currentTarget.value.toLowerCase().trim();
const isValidDomain = isValidENSName(domainToCheck);
this.setState({
domainToCheck,
isValidDomain
});
};
private onSubmit = (ev: React.FormEvent<HTMLElement>) => {
ev.preventDefault();
const { isValidDomain, domainToCheck } = this.state;
return isValidDomain && this.props.resolveDomainRequested(domainToCheck);
};
private onFocus = () => this.setState({ isFocused: true });
private onBlur = () => this.setState({ isFocused: false });
}
function mapStateToProps(state: AppState) {
return {
domainRequests: state.ens.domainRequests
};
}
export default connect(mapStateToProps, {
resolveDomainRequested
})(NameInput);