Skip to content

Commit

Permalink
fix(core): improve error message on reset password page
Browse files Browse the repository at this point in the history
  • Loading branch information
bc-alexsaiannyi committed Oct 29, 2024
1 parent ac83d3e commit 1f3b01d
Show file tree
Hide file tree
Showing 4 changed files with 42 additions and 5 deletions.
5 changes: 5 additions & 0 deletions .changeset/eighty-tomatoes-give.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@bigcommerce/catalyst-core": patch
---

improve error message on reset password page
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,24 @@ const ResetPasswordSchema = z.object({
email: z.string().email(),
});

const processZodErrors = (err: z.ZodError, errPrefix?: string) => {
const fallbackPrefix = 'Validation error in';

const { fieldErrors, formErrors } = err.flatten((issue: z.ZodIssue) => ({
message: issue.message,
}));

if (formErrors.length > 0) {
return formErrors.join('\n');
}

return Object.entries(fieldErrors)
.map(([field, errorList]) => {
return `${errPrefix ?? fallbackPrefix} "${field}": ${errorList?.map(({ message }) => message).join('\n')}`;
})
.join('\n');
};

const ResetPasswordMutation = graphql(`
mutation ResetPassword($input: RequestResetPasswordInput!, $reCaptcha: ReCaptchaV2Input) {
customer {
Expand Down Expand Up @@ -69,7 +87,14 @@ export const resetPassword = async ({
error: result.errors.map((error) => error.message).join('\n'),
};
} catch (error: unknown) {
if (error instanceof Error || error instanceof z.ZodError) {
if (error instanceof z.ZodError) {
return {
status: 'error',
error: processZodErrors(error, t('Errors.validatationError')),
};
}

if (error instanceof Error) {
return { status: 'error', error: error.message };
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ export const ResetPasswordForm = ({ reCaptchaSettings }: Props) => {
};

const handleEmailValidation = (e: ChangeEvent<HTMLInputElement>) => {
const validationStatus = e.target.validity.valueMissing;
const validationStatus = e.target.validity.valueMissing || e.target.validity.typeMismatch;

setIsEmailValid(!validationStatus);
};
Expand Down Expand Up @@ -122,7 +122,7 @@ export const ResetPasswordForm = ({ reCaptchaSettings }: Props) => {
return (
<>
{formStatus?.status === 'error' && (
<Message className="mb-8 w-full" variant={formStatus.status}>
<Message className="mb-8 w-full whitespace-pre" variant={formStatus.status}>
<p>{formStatus.message}</p>
</Message>
)}
Expand All @@ -144,11 +144,17 @@ export const ResetPasswordForm = ({ reCaptchaSettings }: Props) => {
/>
</FieldControl>
<FieldMessage
className="absolute inset-x-0 bottom-0 inline-flex w-full text-xs text-gray-500"
className="absolute inset-x-0 bottom-0 inline-flex w-full text-xs text-error"
match="valueMissing"
>
{t('emailValidationMessage')}
</FieldMessage>
<FieldMessage
className="absolute inset-x-0 bottom-0 inline-flex w-full text-xs text-error"
match="typeMismatch"
>
{t('emailValidationMessage')}
</FieldMessage>
</Field>

{reCaptchaSettings?.isEnabledOnStorefront && (
Expand All @@ -162,7 +168,7 @@ export const ResetPasswordForm = ({ reCaptchaSettings }: Props) => {
sitekey={reCaptchaSettings.siteKey}
/>
{!isReCaptchaValid && (
<span className="absolute inset-x-0 bottom-0 inline-flex w-full text-xs font-normal text-red-200">
<span className="absolute inset-x-0 bottom-0 inline-flex w-full text-xs font-normal text-error">
{t('recaptchaText')}
</span>
)}
Expand Down
1 change: 1 addition & 0 deletions core/messages/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@
"submit": "Reset password"
},
"Errors": {
"validatationError": "Validation error in",
"error":"Unknown error."
}
}
Expand Down

0 comments on commit 1f3b01d

Please sign in to comment.