-
Notifications
You must be signed in to change notification settings - Fork 181
/
login-dialog.tsx
139 lines (117 loc) · 3.6 KB
/
login-dialog.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
import React, {Component, ReactNode} from 'react';
import PropTypes from 'prop-types';
import Dialog from '../dialog/dialog';
import {Content} from '../island/island';
import LoaderScreen from '../loader-screen/loader-screen';
import {HUB_AUTH_PAGE_OPENED} from '../auth/background-flow';
import styles from './login-dialog.css';
const HUB_AUTH_PAGE_LOGIN_STARTED = 'HUB_AUTH_PAGE_LOGIN_STARTED';
const HUB_AUTH_PAGE_LOGIN_DIMENSIONS = 'HUB_AUTH_PAGE_LOGIN_DIMENSIONS';
const DEFAULT_HEIGHT = 517;
const DEFAULT_WIDTH = 333;
const DEFAULT_SHOW_FALLBACK_TIMEOUT = 5000;
export interface LoginDialogProps {
onCancel: () => void
show: boolean,
url: string,
renderFallbackLink: (loggingIn: boolean) => ReactNode,
showFallbackTimeout: number,
className?: string | undefined,
loader?: boolean | null | undefined,
loadingMessage?: string | null | undefined,
}
/**
* @name Login Dialog
*/
export default class LoginDialog extends Component<LoginDialogProps> {
static propTypes = {
show: PropTypes.bool,
className: PropTypes.string,
url: PropTypes.string,
loader: PropTypes.bool,
loadingMessage: PropTypes.string,
showFallbackTimeout: PropTypes.number,
renderFallbackLink: PropTypes.func,
onCancel: PropTypes.func.isRequired
};
static defaultProps = {
show: false,
url: 'about:blank',
renderFallbackLink: () => null,
showFallbackTimeout: DEFAULT_SHOW_FALLBACK_TIMEOUT
};
state = {
loading: true,
loggingIn: false,
showFallbackLink: false,
height: DEFAULT_HEIGHT,
width: DEFAULT_WIDTH
};
componentDidMount() {
window.addEventListener('message', this.onMessage);
this.startFallbackCountdown();
}
componentWillUnmount() {
window.removeEventListener('message', this.onMessage);
}
showFallbackTimout?: number;
startFallbackCountdown() {
this.showFallbackTimout = window.setTimeout(
() => this.setState({showFallbackLink: true}),
this.props.showFallbackTimeout
);
}
onMessage = (event: MessageEvent) => {
const {data} = event;
if (!data) {
return;
}
if (data === HUB_AUTH_PAGE_OPENED) {
clearTimeout(this.showFallbackTimout);
this.setState({loading: false, loggingIn: false});
return;
}
if (data === HUB_AUTH_PAGE_LOGIN_STARTED) {
this.setState({loading: true, loggingIn: true});
this.startFallbackCountdown();
return;
}
if (data.message === HUB_AUTH_PAGE_LOGIN_DIMENSIONS) {
this.setState({height: data.height, width: data.width});
}
};
render() {
const {show, className, url, loadingMessage, renderFallbackLink, onCancel} = this.props;
const {loading, height, width, loggingIn, showFallbackLink} = this.state;
const iFrameStyle = {height, width};
return (
<Dialog
data-test="ring-login-dialog"
className={className}
contentClassName={styles.dialogContent}
trapFocus
autoFocusFirst={false}
show={show}
showCloseButton
onCloseAttempt={onCancel}
>
<Content>
<iframe
title="Login dialog"
style={iFrameStyle}
src={url}
className={styles.iFrame}
scrolling="no"
/>
</Content>
{loading && (
<LoaderScreen message={loadingMessage} containerClassName={styles.nonOpaqueLoader}/>
)}
{showFallbackLink && (
<div className={styles.fallbackLinkContainer}>{renderFallbackLink(loggingIn)}</div>
)}
</Dialog>
);
}
}
export type LoginDialogAttrs = JSX.LibraryManagedAttributes<typeof LoginDialog, LoginDialogProps>