2018-04-06 23:32:25 +02:00
|
|
|
import React from 'react';
|
|
|
|
import classnames from 'classnames';
|
2018-06-17 20:53:00 -05:00
|
|
|
import QrSigner from '@parity/qr-signer';
|
|
|
|
|
2018-04-06 23:32:25 +02:00
|
|
|
import translate from 'translations';
|
|
|
|
import { Spinner } from 'components/ui';
|
|
|
|
import './ParityQrSigner.scss';
|
|
|
|
|
|
|
|
interface State {
|
|
|
|
webcamError: null | React.ReactElement<any>;
|
|
|
|
isLoading: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ScanProps {
|
|
|
|
scan: true;
|
|
|
|
onScan(data: string): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ShowProps {
|
|
|
|
scan: false;
|
|
|
|
account: string;
|
2018-04-26 02:36:29 +02:00
|
|
|
data?: string;
|
|
|
|
rlp?: string;
|
2018-04-06 23:32:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
interface SharedProps {
|
|
|
|
size?: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
type Props = (ScanProps | ShowProps) & SharedProps;
|
|
|
|
|
|
|
|
export default class ParityQrSigner extends React.PureComponent<Props, State> {
|
|
|
|
public state: State = {
|
|
|
|
webcamError: null,
|
|
|
|
isLoading: true
|
|
|
|
};
|
|
|
|
|
|
|
|
public componentDidMount() {
|
|
|
|
this.checkForWebcam();
|
|
|
|
if (navigator.mediaDevices) {
|
|
|
|
navigator.mediaDevices.addEventListener('devicechange', this.checkForWebcam);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public componentWillUnmount() {
|
|
|
|
if (navigator.mediaDevices && navigator.mediaDevices.ondevicechange) {
|
|
|
|
navigator.mediaDevices.removeEventListener('devicechange', this.checkForWebcam);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public checkForWebcam = async () => {
|
|
|
|
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
|
|
|
|
try {
|
|
|
|
await navigator.mediaDevices.getUserMedia({ video: true });
|
|
|
|
this.setState({
|
|
|
|
webcamError: null,
|
|
|
|
isLoading: false
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
const err = e as DOMException;
|
|
|
|
let errorMessage;
|
|
|
|
switch (err.name) {
|
|
|
|
case 'NotAllowedError':
|
|
|
|
case 'SecurityError':
|
|
|
|
errorMessage = translate('ADD_PARITY_ERROR_DISABLED');
|
|
|
|
break;
|
|
|
|
case 'NotFoundError':
|
|
|
|
case 'OverconstrainedError':
|
|
|
|
errorMessage = translate('ADD_PARITY_ERROR_NO_CAM');
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
errorMessage = translate('ADD_PARITY_ERROR_UNKNOWN');
|
|
|
|
}
|
|
|
|
this.setState({
|
|
|
|
webcamError: errorMessage,
|
|
|
|
isLoading: false
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
public render() {
|
|
|
|
const { webcamError, isLoading } = this.state;
|
|
|
|
const size = this.props.size || 300;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className={classnames({
|
|
|
|
ParityQrSigner: true,
|
|
|
|
'is-disabled': !!webcamError || isLoading
|
|
|
|
})}
|
|
|
|
style={{
|
|
|
|
width: size,
|
|
|
|
height: size
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{isLoading ? (
|
|
|
|
<div className="ParityQrSigner-loader">
|
|
|
|
<Spinner size="x3" light={true} />
|
|
|
|
</div>
|
|
|
|
) : webcamError ? (
|
|
|
|
<div className="ParityQrSigner-error">
|
|
|
|
<i className="ParityQrSigner-error-icon fa fa-exclamation-circle" />
|
|
|
|
{webcamError}
|
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<QrSigner {...this.props} size={size} />
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|