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
|
/** @jsx jsx */
import { css, jsx } from "@emotion/react";
import React, { useState } from "react";
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { faDiscord } from "@fortawesome/free-brands-svg-icons";
import authenticate, { APIErrors, checkScopes, OAuthScopes } from "../api/auth";
import { selectable } from "../commonStyles";
interface OAuth2ButtonProps {
scopes?: OAuthScopes[],
rerender?: () => void
}
const iconStyles = css`
position: relative;
top: 0.3rem;
padding-left: 0.65rem;
font-size: 1.2em;
`;
const textStyles = css`
display: inline-block;
padding: 0.5rem 0.75rem 0.5rem 0.5rem;
`;
const errorStyles = css`
position: absolute;
visibility: hidden;
width: 100%;
left: 0;
text-align: center;
white-space: normal;
box-sizing: border-box;
padding: 0 15rem;
@media (max-width: 750px) {
padding: 0 5rem;
}
color: red;
margin-top: 2.5rem;
`;
async function login(props: OAuth2ButtonProps, errorDialog: React.RefObject<HTMLDivElement>, setDisabled: (newState: boolean) => void) {
await authenticate(props.scopes, setDisabled).catch((reason: APIErrors) => {
// Display Error Message
if (errorDialog.current) {
errorDialog.current.style.visibility = "visible";
errorDialog.current.textContent = reason.Message;
errorDialog.current.scrollIntoView({behavior: "smooth"});
}
// Propagate to sentry
reason.Error.stack = new Error(`OAuth: ${reason.Message}`).stack + "\n" + reason.Error.stack;
throw reason.Error;
});
if (checkScopes(props.scopes) && props.rerender) {
props.rerender();
}
}
function OAuth2Button(props: OAuth2ButtonProps): JSX.Element {
const [disabled, setDisabled] = useState<boolean>(false);
const errorDialog: React.RefObject<HTMLDivElement> = React.useRef(null);
return <span>
<button disabled={disabled} onClick={() => login(props, errorDialog, setDisabled)}>
<FontAwesomeIcon icon={faDiscord} css={iconStyles}/>
<span css={textStyles}>Login To Submit</span>
</button>
<div css={[errorStyles, selectable]} ref={errorDialog}/>
</span>;
}
export default OAuth2Button;
|