Update update-email.ftl page

This commit is contained in:
Joseph Garrone 2024-05-08 19:24:18 +02:00
parent ef3c190747
commit 0d36ddd6d3
4 changed files with 98 additions and 87 deletions

View File

@ -338,7 +338,6 @@ function InputTag(props: InputFiledByTypeProps & { fieldIndex: number | undefine
max={attribute.annotations.inputTypeMax} max={attribute.annotations.inputTypeMax}
min={attribute.annotations.inputTypeMin} min={attribute.annotations.inputTypeMin}
step={attribute.annotations.inputTypeStep} step={attribute.annotations.inputTypeStep}
// NOTE: The `?? {}` is for backward compatibility with Keycloak prior to 24
{...Object.fromEntries(Object.entries(attribute.html5DataAnnotations ?? {}).map(([key, value]) => [`data-${key}`, value]))} {...Object.fromEntries(Object.entries(attribute.html5DataAnnotations ?? {}).map(([key, value]) => [`data-${key}`, value]))}
onChange={event => onChange={event =>
formValidationDispatch({ formValidationDispatch({

View File

@ -175,11 +175,7 @@ export declare namespace KcContext {
export type Register = Common & { export type Register = Common & {
pageId: "register.ftl" | "register-user-profile.ftl"; pageId: "register.ftl" | "register-user-profile.ftl";
profile: { profile: UserProfile;
attributes: Attribute[];
attributesByName: Record<string, Attribute>;
html5DataAnnotations?: Record<string, string>;
};
url: { url: {
registrationAction: string; registrationAction: string;
}; };
@ -423,39 +419,17 @@ export declare namespace KcContext {
export type LoginUpdateProfile = Common & { export type LoginUpdateProfile = Common & {
pageId: "login-update-profile.ftl" | "update-user-profile.ftl"; pageId: "login-update-profile.ftl" | "update-user-profile.ftl";
profile: { profile: UserProfile;
attributes: Attribute[];
attributesByName: Record<string, Attribute>;
}; };
};
/*
export type LoginUpdateProfile = Common & {
pageId: "login-update-profile.ftl";
user: {
editUsernameAllowed: boolean;
username?: string;
email?: string;
firstName?: string;
lastName?: string;
};
};
*/
export type IdpReviewUserProfile = Common & { export type IdpReviewUserProfile = Common & {
pageId: "idp-review-user-profile.ftl"; pageId: "idp-review-user-profile.ftl";
profile: { profile: UserProfile;
context: "IDP_REVIEW";
attributes: Attribute[];
attributesByName: Record<string, Attribute>;
};
}; };
export type UpdateEmail = Common & { export type UpdateEmail = Common & {
pageId: "update-email.ftl"; pageId: "update-email.ftl";
email: { profile: UserProfile;
value?: string;
};
}; };
export type SelectAuthenticator = Common & { export type SelectAuthenticator = Common & {
@ -497,6 +471,12 @@ export declare namespace KcContext {
}; };
} }
export type UserProfile = {
attributes: Attribute[];
attributesByName: Record<string, Attribute>;
html5DataAnnotations?: Record<string, string>;
};
export type Attribute = { export type Attribute = {
name: string; name: string;
displayName?: string; displayName?: string;

View File

@ -151,7 +151,7 @@ export function useUserProfileForm(params: ParamsOfUseUserProfileForm): ReturnTy
"name": name, "name": name,
"displayName": id<`\${${MessageKey}}`>(`\${${name}}`), "displayName": id<`\${${MessageKey}}`>(`\${${name}}`),
"required": true, "required": true,
"value": (kcContext as any).register.formData[name] ?? "", "value": (kcContext.register as any).formData[name] ?? "",
"html5DataAnnotations": {}, "html5DataAnnotations": {},
"readOnly": false, "readOnly": false,
"validators": {}, "validators": {},
@ -173,7 +173,7 @@ export function useUserProfileForm(params: ParamsOfUseUserProfileForm): ReturnTy
if ("user" in kcContext && kcContext.user instanceof Object) { if ("user" in kcContext && kcContext.user instanceof Object) {
//NOTE: Handle legacy login-update-profile.ftl //NOTE: Handle legacy login-update-profile.ftl
return (["username", "email", "firstName", "lastName"] as const) return (["username", "email", "firstName", "lastName"] as const)
.filter(name => (name !== "username" ? true : (kcContext as any).user.editUsernameAllowed)) .filter(name => (name !== "username" ? true : (kcContext.user as any).editUsernameAllowed))
.map(name => .map(name =>
id<Attribute>({ id<Attribute>({
"name": name, "name": name,
@ -198,6 +198,23 @@ export function useUserProfileForm(params: ParamsOfUseUserProfileForm): ReturnTy
); );
} }
if ("email" in kcContext && kcContext.email instanceof Object) {
//NOTE: Handle legacy update-email.ftl
return [
id<Attribute>({
"name": "email",
"displayName": id<`\${${MessageKey}}`>(`\${email}`),
"required": true,
"value": (kcContext.email as any).value ?? "",
"html5DataAnnotations": {},
"readOnly": false,
"validators": {},
"annotations": {},
"autocomplete": "email"
})
];
}
assert(false, "Unable to mock user profile from the current kcContext"); assert(false, "Unable to mock user profile from the current kcContext");
} }

View File

@ -1,11 +1,18 @@
import { useState } from "react";
import { clsx } from "keycloakify/tools/clsx"; import { clsx } from "keycloakify/tools/clsx";
import type { LazyOrNot } from "keycloakify/tools/LazyOrNot";
import type { UserProfileFormFieldsProps } from "keycloakify/login/UserProfileFormFields";
import type { PageProps } from "keycloakify/login/pages/PageProps"; import type { PageProps } from "keycloakify/login/pages/PageProps";
import { useGetClassName } from "keycloakify/login/lib/useGetClassName"; import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
import type { KcContext } from "../kcContext"; import type { KcContext } from "../kcContext";
import type { I18n } from "../i18n"; import type { I18n } from "../i18n";
export default function UpdateEmail(props: PageProps<Extract<KcContext, { pageId: "update-email.ftl" }>, I18n>) { type UpdateEmailProps = PageProps<Extract<KcContext, { pageId: "update-email.ftl" }>, I18n> & {
const { kcContext, i18n, doUseDefaultCss, Template, classes } = props; UserProfileFormFields: LazyOrNot<(props: UserProfileFormFieldsProps) => JSX.Element>;
};
export default function UpdateEmail(props: UpdateEmailProps) {
const { kcContext, i18n, doUseDefaultCss, Template, classes, UserProfileFormFields } = props;
const { getClassName } = useGetClassName({ const { getClassName } = useGetClassName({
doUseDefaultCss, doUseDefaultCss,
@ -14,47 +21,48 @@ export default function UpdateEmail(props: PageProps<Extract<KcContext, { pageId
const { msg, msgStr } = i18n; const { msg, msgStr } = i18n;
const { url, messagesPerField, isAppInitiatedAction, email } = kcContext; const [isFormSubmittable, setIsFormSubmittable] = useState(false);
const { url, messagesPerField, isAppInitiatedAction } = kcContext;
return ( return (
<Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("updateEmailTitle")}> <Template
<form id="kc-update-email-form" className={getClassName("kcFormClass")} action={url.loginAction} method="post"> {...{ kcContext, i18n, doUseDefaultCss, classes }}
<div displayMessage={messagesPerField.exists("global")}
className={clsx(getClassName("kcFormGroupClass"), messagesPerField.printIfExists("email", getClassName("kcFormGroupErrorClass")))} displayRequiredFields={true}
headerNode={msg("updateEmailTitle")}
> >
<div className={getClassName("kcLabelWrapperClass")}> <form id="kc-update-email-form" className={getClassName("kcFormClass")} action={url.loginAction} method="post">
<label htmlFor="email" className={getClassName("kcLabelClass")}> <UserProfileFormFields
{msg("email")} {...{
</label> kcContext,
</div> i18n,
<div className={getClassName("kcInputWrapperClass")}> getClassName,
<input messagesPerField
type="text" }}
id="email" onIsFormSubmittableValueChange={setIsFormSubmittable}
name="email"
defaultValue={email.value ?? ""}
className={getClassName("kcInputClass")}
aria-invalid={messagesPerField.existsError("email")}
/> />
</div>
</div>
<div className={getClassName("kcFormGroupClass")}> <div className={getClassName("kcFormGroupClass")}>
<div id="kc-form-options" className={getClassName("kcFormOptionsClass")}> <div id="kc-form-options" className={getClassName("kcFormOptionsClass")}>
<div className={getClassName("kcFormOptionsWrapperClass")}></div> <div className={getClassName("kcFormOptionsWrapperClass")} />
</div> </div>
<LogoutOtherSessions {...{ getClassName, i18n }} />
<div id="kc-form-buttons" className={getClassName("kcFormButtonsClass")}> <div id="kc-form-buttons" className={getClassName("kcFormButtonsClass")}>
{isAppInitiatedAction ? (
<>
<input <input
disabled={!isFormSubmittable}
className={clsx( className={clsx(
getClassName("kcButtonClass"), getClassName("kcButtonClass"),
getClassName("kcButtonPrimaryClass"), getClassName("kcButtonPrimaryClass"),
isAppInitiatedAction && getClassName("kcButtonBlockClass"),
getClassName("kcButtonLargeClass") getClassName("kcButtonLargeClass")
)} )}
type="submit" type="submit"
defaultValue={msgStr("doSubmit")} value={msgStr("doSubmit")}
/> />
{isAppInitiatedAction && (
<button <button
className={clsx( className={clsx(
getClassName("kcButtonClass"), getClassName("kcButtonClass"),
@ -67,18 +75,6 @@ export default function UpdateEmail(props: PageProps<Extract<KcContext, { pageId
> >
{msg("doCancel")} {msg("doCancel")}
</button> </button>
</>
) : (
<input
className={clsx(
getClassName("kcButtonClass"),
getClassName("kcButtonPrimaryClass"),
getClassName("kcButtonBlockClass"),
getClassName("kcButtonLargeClass")
)}
type="submit"
defaultValue={msgStr("doSubmit")}
/>
)} )}
</div> </div>
</div> </div>
@ -86,3 +82,22 @@ export default function UpdateEmail(props: PageProps<Extract<KcContext, { pageId
</Template> </Template>
); );
} }
function LogoutOtherSessions(props: { getClassName: ReturnType<typeof useGetClassName>["getClassName"]; i18n: I18n }) {
const { getClassName, i18n } = props;
const { msg } = i18n;
return (
<div id="kc-form-options" className={getClassName("kcFormOptionsClass")}>
<div className={getClassName("kcFormOptionsWrapperClass")}>
<div className="checkbox">
<label>
<input type="checkbox" id="logout-sessions" name="logout-sessions" value="on" defaultChecked={true} />
{msg("logoutOtherSessions")}
</label>
</div>
</div>
</div>
);
}