1
0
mirror of https://github.com/0rangebananaspy/authelia.git synced 2024-09-14 22:47:21 +07:00
authelia/web/src/views/LoginPortal/ConsentView/ConsentView.tsx

283 lines
9.8 KiB
TypeScript

import React, { useEffect, Fragment, ReactNode, useState } from "react";
import { AccountBox, CheckBox, Contacts, Drafts, Group } from "@mui/icons-material";
import {
Button,
Grid,
List,
ListItem,
ListItemIcon,
ListItemText,
Tooltip,
Typography,
Checkbox,
FormControlLabel,
Theme,
} from "@mui/material";
import makeStyles from "@mui/styles/makeStyles";
import { useTranslation } from "react-i18next";
import { useNavigate } from "react-router-dom";
import { IndexRoute } from "@constants/Routes";
import { useConsentResponse } from "@hooks/Consent";
import { useNotifications } from "@hooks/NotificationsContext";
import { useRedirector } from "@hooks/Redirector";
import { useUserInfoGET } from "@hooks/UserInfo";
import LoginLayout from "@layouts/LoginLayout";
import { acceptConsent, rejectConsent } from "@services/Consent";
import LoadingPage from "@views/LoadingPage/LoadingPage";
export interface Props {}
function scopeNameToAvatar(id: string) {
switch (id) {
case "openid":
return <AccountBox />;
case "profile":
return <Contacts />;
case "groups":
return <Group />;
case "email":
return <Drafts />;
default:
return <CheckBox />;
}
}
const ConsentView = function (props: Props) {
const styles = useStyles();
const navigate = useNavigate();
const redirect = useRedirector();
const { createErrorNotification, resetNotification } = useNotifications();
const [resp, fetch, , err] = useConsentResponse();
const { t: translate } = useTranslation();
const [preConfigure, setPreConfigure] = useState(false);
const handlePreConfigureChanged = () => {
setPreConfigure((preConfigure) => !preConfigure);
};
const [userInfo, fetchUserInfo, , fetchUserInfoError] = useUserInfoGET();
useEffect(() => {
fetchUserInfo();
}, [fetchUserInfo]);
useEffect(() => {
if (fetchUserInfoError) {
createErrorNotification("There was an issue retrieving user preferences");
}
}, [fetchUserInfoError, createErrorNotification]);
useEffect(() => {
if (err) {
navigate(IndexRoute);
console.error(`Unable to display consent screen: ${err.message}`);
}
}, [navigate, resetNotification, createErrorNotification, err]);
useEffect(() => {
fetch();
}, [fetch]);
const translateScopeNameToDescription = (id: string): string => {
switch (id) {
case "openid":
return translate("Use OpenID to verify your identity");
case "profile":
return translate("Access your profile information");
case "groups":
return translate("Access your group membership");
case "email":
return translate("Access your email addresses");
default:
return id;
}
};
const handleAcceptConsent = async () => {
// This case should not happen in theory because the buttons are disabled when response is undefined.
if (!resp) {
return;
}
const res = await acceptConsent(resp.client_id, preConfigure);
if (res.redirect_uri) {
redirect(res.redirect_uri);
} else {
throw new Error("Unable to redirect the user");
}
};
const handleRejectConsent = async () => {
if (!resp) {
return;
}
const res = await rejectConsent(resp.client_id);
if (res.redirect_uri) {
redirect(res.redirect_uri);
} else {
throw new Error("Unable to redirect the user");
}
};
return (
<ComponentOrLoading ready={resp !== undefined && userInfo !== undefined}>
<LoginLayout
id="consent-stage"
title={`${translate("Hi")} ${userInfo?.display_name}`}
subtitle={translate("Consent Request")}
showBrand
>
<Grid container>
<Grid item xs={12}>
<div>
<Tooltip
title={
translate("Client ID", { client_id: resp?.client_id }) ||
"Client ID: " + resp?.client_id
}
>
<Typography className={styles.clientDescription}>
{resp !== undefined && resp.client_description !== ""
? resp.client_description
: resp?.client_id}
</Typography>
</Tooltip>
</div>
</Grid>
<Grid item xs={12}>
<div>{translate("The above application is requesting the following permissions")}:</div>
</Grid>
<Grid item xs={12}>
<div className={styles.scopesListContainer}>
<List className={styles.scopesList}>
{resp?.scopes.map((scope: string) => (
<Tooltip title={"Scope " + scope}>
<ListItem id={"scope-" + scope} dense>
<ListItemIcon>{scopeNameToAvatar(scope)}</ListItemIcon>
<ListItemText primary={translateScopeNameToDescription(scope)} />
</ListItem>
</Tooltip>
))}
</List>
</div>
</Grid>
{resp?.pre_configuration ? (
<Grid item xs={12}>
<Tooltip
title={
translate("This saves this consent as a pre-configured consent for future use") ||
"This saves this consent as a pre-configured consent for future use"
}
>
<FormControlLabel
control={
<Checkbox
id="pre-configure"
checked={preConfigure}
onChange={handlePreConfigureChanged}
value="preConfigure"
color="primary"
/>
}
className={styles.preConfigure}
label={translate("Remember Consent")}
/>
</Tooltip>
</Grid>
) : null}
<Grid item xs={12}>
<Grid container spacing={1}>
<Grid item xs={6}>
<Button
id="accept-button"
className={styles.button}
disabled={!resp}
onClick={handleAcceptConsent}
color="primary"
variant="contained"
>
{translate("Accept")}
</Button>
</Grid>
<Grid item xs={6}>
<Button
id="deny-button"
className={styles.button}
disabled={!resp}
onClick={handleRejectConsent}
color="secondary"
variant="contained"
>
{translate("Deny")}
</Button>
</Grid>
</Grid>
</Grid>
</Grid>
</LoginLayout>
</ComponentOrLoading>
);
};
const useStyles = makeStyles((theme: Theme) => ({
container: {
paddingTop: theme.spacing(4),
paddingBottom: theme.spacing(4),
display: "block",
justifyContent: "center",
},
clientDescription: {
fontWeight: 600,
},
scopesListContainer: {
textAlign: "center",
},
scopesList: {
display: "inline-block",
backgroundColor: theme.palette.background.paper,
marginTop: theme.spacing(2),
marginBottom: theme.spacing(2),
},
clientID: {
fontWeight: "bold",
},
button: {
marginLeft: theme.spacing(),
marginRight: theme.spacing(),
width: "100%",
},
bulletIcon: {
display: "inline-block",
},
permissionsContainer: {
border: "1px solid #dedede",
margin: theme.spacing(4),
},
listItem: {
textAlign: "center",
marginRight: theme.spacing(2),
},
preConfigure: {},
}));
export default ConsentView;
interface ComponentOrLoadingProps {
ready: boolean;
children: ReactNode;
}
function ComponentOrLoading(props: ComponentOrLoadingProps) {
return (
<Fragment>
<div className={props.ready ? "hidden" : ""}>
<LoadingPage />
</div>
{props.ready ? props.children : null}
</Fragment>
);
}