Mailspring/internal_packages/onboarding/lib/page-authenticate.jsx

177 lines
5.3 KiB
JavaScript

import React from 'react';
import {shell} from 'electron'
import classnames from 'classnames';
import ReactDOM from 'react-dom';
import {IdentityStore} from 'nylas-exports';
import {RetinaImg} from 'nylas-component-kit';
import networkErrors from 'chromium-net-errors';
import OnboardingActions from './onboarding-actions';
class InitialLoadingCover extends React.Component {
static propTypes = {
ready: React.PropTypes.bool,
error: React.PropTypes.string,
onTryAgain: React.PropTypes.func,
}
constructor(props) {
super(props);
this.state = {};
}
componentDidMount() {
this._slowTimeout = setTimeout(() => {
this.setState({slow: true});
}, 3500);
}
componentWillUnmount() {
clearTimeout(this._slowTimeout);
this._slowTimeout = null;
}
render() {
const classes = classnames({
'webview-cover': true,
'ready': this.props.ready,
'error': this.props.error,
'slow': this.state.slow,
});
let message = this.props.error;
if (this.props.error) {
message = this.props.error;
} else if (this.state.slow) {
message = "Still trying to reach Nylas…";
} else {
message = ' '
}
return (
<div className={classes}>
<div style={{flex: 1}} />
<RetinaImg
className="spinner"
style={{width: 20, height: 20}}
name="inline-loading-spinner.gif"
mode={RetinaImg.Mode.ContentPreserve}
/>
<div className="message">{message}</div>
<div className="btn try-again" onClick={this.props.onTryAgain}>Try Again</div>
<div style={{flex: 1}} />
</div>
);
}
}
export default class AuthenticatePage extends React.Component {
static displayName = "AuthenticatePage";
static propTypes = {
accountInfo: React.PropTypes.object,
};
constructor(props) {
super(props);
this.state = {
ready: false,
error: null,
};
}
componentDidMount() {
const webview = ReactDOM.findDOMNode(this.refs.webview);
const n1Version = NylasEnv.getVersion();
webview.partition = 'in-memory-only';
webview.src = `${IdentityStore.URLRoot}/onboarding?utm_medium=N1&utm_source=OnboardingPage&N1_version=${n1Version}&client_edition=basic`;
webview.addEventListener('did-start-loading', this.webviewDidStartLoading);
webview.addEventListener('did-get-response-details', this.webviewDidGetResponseDetails);
webview.addEventListener('did-fail-load', this.webviewDidFailLoad);
webview.addEventListener('did-finish-load', this.webviewDidFinishLoad);
webview.addEventListener('console-message', (e) => {
if (/^http.+/i.test(e.message)) { shell.openExternal(e.message) }
console.log('Guest page logged a message:', e.message);
});
}
onTryAgain = () => {
const webview = ReactDOM.findDOMNode(this.refs.webview);
webview.reload();
}
webviewDidStartLoading = () => {
this.setState({error: null, webviewLoading: true});
}
webviewDidGetResponseDetails = ({httpResponseCode, originalURL}) => {
if (!originalURL.includes(IdentityStore.URLRoot)) {
// This means that some other secondarily loaded resource (like
// analytics or Linkedin, etc) got a response. We don't care about
// that.
return
}
if (httpResponseCode >= 400) {
const error = `
Could not reach Nylas to sign in. Please try again or contact
support@nylas.com if the issue persists.
(${originalURL}: ${httpResponseCode})
`;
this.setState({ready: false, error: error, webviewLoading: false});
}
};
webviewDidFailLoad = ({errorCode, validatedURL}) => {
// "Operation was aborted" can be fired when we move between pages quickly.
if (errorCode === -3) {
return;
}
const e = networkErrors.createByCode(errorCode);
const error = `Could not reach ${validatedURL}. ${e ? e.message : errorCode}`;
this.setState({ready: false, error: error, webviewLoading: false});
}
webviewDidFinishLoad = () => {
// this is sometimes called right after did-fail-load
if (this.state.error) return;
const webview = ReactDOM.findDOMNode(this.refs.webview);
const receiveUserInfo = `
var a = document.querySelector('#pro-account');
result = a ? a.innerText : null;
`;
webview.executeJavaScript(receiveUserInfo, false, (result) => {
this.setState({ready: true, webviewLoading: false});
if (result !== null) {
OnboardingActions.authenticationJSONReceived(JSON.parse(result));
}
});
const openExternalLink = `
var el = document.querySelector('.open-external');
if (el) {el.addEventListener('click', function(event) {console.log(this.href); event.preventDefault(); return false;})}
`;
webview.executeJavaScript(openExternalLink);
}
render() {
return (
<div className="page authenticate">
<webview ref="webview" is partition="in-memory-only" />
<div className={`webview-loading-spinner loading-${this.state.webviewLoading}`}>
<RetinaImg
style={{width: 20, height: 20}}
name="inline-loading-spinner.gif"
mode={RetinaImg.Mode.ContentPreserve}
/>
</div>
<InitialLoadingCover
ready={this.state.ready}
error={this.state.error}
onTryAgain={this.onTryAgain}
/>
</div>
);
}
}