feat(web): add user display name to oidc consent view (#3138)

This adds the current logged in users display name to the consent page as well as some other minor tweaks.

Closes #2595
pull/3139/head
James Elliott 2022-04-08 12:50:55 +10:00 committed by GitHub
parent 5f51dcdb51
commit 90edf11b88
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 108 additions and 20 deletions

View File

@ -61,5 +61,7 @@
"Must have at least one number": "Must have at least one number", "Must have at least one number": "Must have at least one number",
"Must have at least one special character": "Must have at least one special character", "Must have at least one special character": "Must have at least one special character",
"Must be at least {{len}} characters in length": "Must be at least {{len}} characters in length", "Must be at least {{len}} characters in length": "Must be at least {{len}} characters in length",
"Must not be more than {{len}} characters in length": "Must not be more than {{len}} characters in length" "Must not be more than {{len}} characters in length": "Must not be more than {{len}} characters in length",
"Consent Request": "Consent Request",
"Client ID": "Client ID: {{client_id}}"
} }

View File

@ -0,0 +1,13 @@
import React from "react";
import { render } from "@testing-library/react";
import TypographyWithTooltip from "@components/TypographyWithTootip";
it("renders without crashing", () => {
render(<TypographyWithTooltip value={"Example"} variant={"h5"} />);
});
it("renders with tooltip without crashing", () => {
render(<TypographyWithTooltip value={"Example"} tooltip={"A tooltip"} variant={"h5"} />);
});

View File

@ -0,0 +1,33 @@
import React, { Fragment } from "react";
import { Tooltip, Typography } from "@material-ui/core";
import { Variant } from "@material-ui/core/styles/createTypography";
import { CSSProperties } from "@material-ui/styles";
export interface Props {
variant: Variant;
value?: string;
style?: CSSProperties;
tooltip?: string;
tooltipStyle?: CSSProperties;
}
export default function TypographyWithTooltip(props: Props): JSX.Element {
return (
<Fragment>
{props.tooltip ? (
<Tooltip title={props.tooltip} style={props.tooltipStyle}>
<Typography variant={props.variant} style={props.style}>
{props.value}
</Typography>
</Tooltip>
) : (
<Typography variant={props.variant} style={props.style}>
{props.value}
</Typography>
)}
</Fragment>
);
}

View File

@ -1,6 +1,10 @@
import { useRemoteCall } from "@hooks/RemoteCall"; import { useRemoteCall } from "@hooks/RemoteCall";
import { postUserInfo } from "@services/UserInfo"; import { getUserInfo, postUserInfo } from "@services/UserInfo";
export function useUserInfoPOST() { export function useUserInfoPOST() {
return useRemoteCall(postUserInfo, []); return useRemoteCall(postUserInfo, []);
} }
export function useUserInfoGET() {
return useRemoteCall(getUserInfo, []);
}

View File

@ -1,15 +1,19 @@
import React, { ReactNode } from "react"; import React, { ReactNode } from "react";
import { Grid, makeStyles, Container, Typography, Link } from "@material-ui/core"; import { Grid, makeStyles, Container, Link } from "@material-ui/core";
import { grey } from "@material-ui/core/colors"; import { grey } from "@material-ui/core/colors";
import { ReactComponent as UserSvg } from "@assets/images/user.svg"; import { ReactComponent as UserSvg } from "@assets/images/user.svg";
import TypographyWithTooltip from "@components/TypographyWithTootip";
import { getLogoOverride } from "@utils/Configuration"; import { getLogoOverride } from "@utils/Configuration";
export interface Props { export interface Props {
id?: string; id?: string;
children?: ReactNode; children?: ReactNode;
title?: string; title?: string;
titleTooltip?: string;
subtitle?: string;
subtitleTooltip?: string;
showBrand?: boolean; showBrand?: boolean;
} }
@ -29,9 +33,16 @@ const LoginLayout = function (props: Props) {
</Grid> </Grid>
{props.title ? ( {props.title ? (
<Grid item xs={12}> <Grid item xs={12}>
<Typography variant="h5" className={style.title}> <TypographyWithTooltip variant={"h5"} value={props.title} tooltip={props.titleTooltip} />
{props.title} </Grid>
</Typography> ) : null}
{props.subtitle ? (
<Grid item xs={12}>
<TypographyWithTooltip
variant={"h6"}
value={props.subtitle}
tooltip={props.subtitleTooltip}
/>
</Grid> </Grid>
) : null} ) : null}
<Grid item xs={12} className={style.body}> <Grid item xs={12} className={style.body}>
@ -66,6 +77,7 @@ const useStyles = makeStyles((theme) => ({
paddingRight: 32, paddingRight: 32,
}, },
title: {}, title: {},
subtitle: {},
icon: { icon: {
margin: theme.spacing(), margin: theme.spacing(),
width: "64px", width: "64px",

View File

@ -1,7 +1,7 @@
import { SecondFactorMethod } from "@models/Methods"; import { SecondFactorMethod } from "@models/Methods";
import { UserInfo } from "@models/UserInfo"; import { UserInfo } from "@models/UserInfo";
import { UserInfo2FAMethodPath, UserInfoPath } from "@services/Api"; import { UserInfo2FAMethodPath, UserInfoPath } from "@services/Api";
import { Post, PostWithOptionalResponse } from "@services/Client"; import { Get, Post, PostWithOptionalResponse } from "@services/Client";
export type Method2FA = "webauthn" | "totp" | "mobile_push"; export type Method2FA = "webauthn" | "totp" | "mobile_push";
@ -44,6 +44,11 @@ export async function postUserInfo(): Promise<UserInfo> {
return { ...res, method: toEnum(res.method) }; return { ...res, method: toEnum(res.method) };
} }
export async function getUserInfo(): Promise<UserInfo> {
const res = await Get<UserInfoPayload>(UserInfoPath);
return { ...res, method: toEnum(res.method) };
}
export function setPreferred2FAMethod(method: SecondFactorMethod) { export function setPreferred2FAMethod(method: SecondFactorMethod) {
return PostWithOptionalResponse(UserInfo2FAMethodPath, { method: toString(method) } as MethodPreferencePayload); return PostWithOptionalResponse(UserInfo2FAMethodPath, { method: toString(method) } as MethodPreferencePayload);
} }

View File

@ -19,6 +19,7 @@ import { IndexRoute } from "@constants/Routes";
import { useRequestedScopes } from "@hooks/Consent"; import { useRequestedScopes } from "@hooks/Consent";
import { useNotifications } from "@hooks/NotificationsContext"; import { useNotifications } from "@hooks/NotificationsContext";
import { useRedirector } from "@hooks/Redirector"; import { useRedirector } from "@hooks/Redirector";
import { useUserInfoGET } from "@hooks/UserInfo";
import LoginLayout from "@layouts/LoginLayout"; import LoginLayout from "@layouts/LoginLayout";
import { acceptConsent, rejectConsent } from "@services/Consent"; import { acceptConsent, rejectConsent } from "@services/Consent";
import LoadingPage from "@views/LoadingPage/LoadingPage"; import LoadingPage from "@views/LoadingPage/LoadingPage";
@ -48,6 +49,18 @@ const ConsentView = function (props: Props) {
const [resp, fetch, , err] = useRequestedScopes(); const [resp, fetch, , err] = useRequestedScopes();
const { t: translate } = useTranslation(); const { t: translate } = useTranslation();
const [userInfo, fetchUserInfo, , fetchUserInfoError] = useUserInfoGET();
useEffect(() => {
fetchUserInfo();
}, [fetchUserInfo]);
useEffect(() => {
if (fetchUserInfoError) {
createErrorNotification("There was an issue retrieving user preferences");
}
}, [fetchUserInfoError, createErrorNotification]);
useEffect(() => { useEffect(() => {
if (err) { if (err) {
navigate(IndexRoute); navigate(IndexRoute);
@ -100,22 +113,28 @@ const ConsentView = function (props: Props) {
}; };
return ( return (
<ComponentOrLoading ready={resp !== undefined}> <ComponentOrLoading ready={resp !== undefined && userInfo !== undefined}>
<LoginLayout id="consent-stage" title={`Permissions Request`} showBrand> <LoginLayout
id="consent-stage"
title={`${translate("Hi")} ${userInfo?.display_name}`}
subtitle={translate("Consent Request")}
showBrand
>
<Grid container> <Grid container>
<Grid item xs={12}> <Grid item xs={12}>
<div> <div>
{resp !== undefined && resp.client_description !== "" ? ( <Tooltip
<Tooltip title={"Client ID: " + resp.client_id}> title={
<Typography className={classes.clientDescription}> translate("Client ID", { client_id: resp?.client_id }) ||
{resp.client_description} "Client ID: " + resp?.client_id
</Typography> }
</Tooltip> >
) : ( <Typography className={classes.clientDescription}>
<Tooltip title={"Client ID: " + resp?.client_id}> {resp !== undefined && resp.client_description !== ""
<Typography className={classes.clientDescription}>{resp?.client_id}</Typography> ? resp.client_description
</Tooltip> : resp?.client_id}
)} </Typography>
</Tooltip>
</div> </div>
</Grid> </Grid>
<Grid item xs={12}> <Grid item xs={12}>