Forgot Password

Installation

Add the following Soul components

The forgot-password component uses the button, input and form-status components. Make sure you have added them to your project.

Install the following dependencies

npm install @conform-to/zod @conform-to/react

Copy and paste the following code into your project

sections/forgot-password/index.tsx

import { ForgotPasswordAction, ForgotPasswordForm } from './forgot-password-form';export interface ForgotPasswordProps {  title?: string;  subtitle?: string;  action: ForgotPasswordAction;  emailLabel?: string;  submitLabel?: string;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { *   --forgot-password-font-family: var(--font-family-body); *   --forgot-password-title-font-family: var(--font-family-heading); *   --forgot-password-title: hsl(var(--foreground)); *   --forgot-password-subtitle: hsl(var(--contrast-500)) * } * ``` */export function ForgotPassword({  title = 'Forgot your password?',  subtitle = 'Enter the email associated with your account below. We’ll send you instructions to reset your password.',  emailLabel,  submitLabel,  action,}: ForgotPasswordProps) {  return (    <div className="@container">      <div className="flex flex-col justify-center gap-y-24 px-3 py-10 @xl:flex-row @xl:px-6 @4xl:py-20 @5xl:px-20">        <div className="flex w-full flex-col @xl:max-w-md @xl:pr-10 @4xl:pr-20">          <header className="font-[family-name:var(--forgot-password-font-family,var(--font-family-body))]">            <h1 className="mb-5 font-[family-name:var(--forgot-password-title-font-family,var(--font-family-heading))] text-4xl font-medium leading-none text-[var(--forgot-password-title,hsl(var(--foreground)))] @xl:text-5xl">              {title}            </h1>            <p className="mb-10 text-base font-light leading-none text-[var(--forgot-password-subtitle,hsl(var(--contrast-500)))] @xl:text-lg">              {subtitle}            </p>          </header>          <ForgotPasswordForm action={action} emailLabel={emailLabel} submitLabel={submitLabel} />        </div>      </div>    </div>  );}

sections/forgot-password/forgot-password-form.tsx

'use client';import { getFormProps, getInputProps, SubmissionResult, useForm } from '@conform-to/react';import { getZodConstraint, parseWithZod } from '@conform-to/zod';import { useActionState } from 'react';import { ReactNode } from 'react';import { useFormStatus } from 'react-dom';import { FormStatus } from '@/vibes/soul/form/form-status';import { Input } from '@/vibes/soul/form/input';import { Button } from '@/vibes/soul/primitives/button';import { schema } from './schema';type Action<State, Payload> = (state: Awaited<State>, payload: Payload) => State | Promise<State>;export type ForgotPasswordAction = Action<  { lastResult: SubmissionResult | null; successMessage?: string },  FormData>;export interface ForgotPasswordFormProps {  action: ForgotPasswordAction;  emailLabel?: string;  submitLabel?: string;}export function ForgotPasswordForm({  action,  emailLabel = 'Email',  submitLabel = 'Reset password',}: ForgotPasswordFormProps) {  const [{ lastResult, successMessage }, formAction] = useActionState(action, { lastResult: null });  const [form, fields] = useForm({    lastResult,    constraint: getZodConstraint(schema),    shouldValidate: 'onBlur',    shouldRevalidate: 'onInput',    onValidate({ formData }) {      return parseWithZod(formData, { schema });    },  });  return (    <form {...getFormProps(form)} action={formAction} className="flex flex-grow flex-col gap-5">      <Input        {...getInputProps(fields.email, { type: 'text' })}        errors={fields.email.errors}        key={fields.email.id}        label={emailLabel}      />      <SubmitButton>{submitLabel}</SubmitButton>      {form.errors?.map((error, index) => (        <FormStatus key={index} type="error">          {error}        </FormStatus>      ))}      {form.status === 'success' && successMessage != null && (        <FormStatus>{successMessage}</FormStatus>      )}    </form>  );}function SubmitButton({ children }: { children: ReactNode }) {  const { pending } = useFormStatus();  return (    <Button className="mt-auto w-full" loading={pending} type="submit" variant="secondary">      {children}    </Button>  );}

sections/forgot-password/schema.ts

import { z } from 'zod';export const schema = z.object({  email: z.string().email({ message: 'Please enter a valid email.' }).trim(),});

Usage

import { ForgotPassword } from '@/vibes/soul/sections/forgot-password';import { schema } from '@/vibes/soul/sections/forgot-password/schema';import { SubmissionResult } from '@conform-to/react';import { parseWithZod } from '@conform-to/zod';function Usage() {  return <ForgotPassword action={forgotPasswordAction} />;}async function forgotPasswordAction(state: { lastResult: SubmissionResult | null; successMessage?: string },formData: FormData,): Promise<{ lastResult: SubmissionResult | null; successMessage?: string }> {  'use server';  const submission = parseWithZod(formData, { schema });  if (submission.status !== 'success') {      return { lastResult: submission.reply({ formErrors: ['Boom!'] }) };  }  // Simulate a network request  await new Promise((resolve) => setTimeout(resolve, 1000));  return {      lastResult: submission.reply(),      successMessage: 'Check your email for a reset link',  };}const schema = z.object({  email: z.string().email({ message: 'Please enter a valid email.' }).trim(),});

API Reference

ForgotPasswordProps

PropTypeDefault
title
string
'Forgot your password?'
subtitle
string
'Enter the email associated with your account below. We’ll send you instructions to reset your password.'
action*
ForgotPasswordAction
emailLabel
string
submitLabel
string

ForgotPasswordAction

type Action<State, Payload> = (state: Awaited<State>, payload: Payload) => State | Promise<State>;export type ForgotPasswordAction = Action<  { lastResult: SubmissionResult | null; successMessage?: string },  FormData>;

This component uses Confom to handle form submissions. Refer to the Conform docs for more details.

Here's an example of an action function that does validation and simulates sending an email:

import { schema } from '@/vibes/soul/sections/forgot-password/schema';import { SubmissionResult } from '@conform-to/react';import { parseWithZod } from '@conform-to/zod';async function forgotPasswordAction(  state: { lastResult: SubmissionResult | null; successMessage?: string },  formData: FormData,): Promise<{ lastResult: SubmissionResult | null; successMessage?: string }> {  'use server';  const submission = parseWithZod(formData, { schema });  if (submission.status !== 'success') {    return { lastResult: submission.reply({ formErrors: ['Boom!'] }) };  }  // Simulate sending email  await new Promise((resolve) => setTimeout(resolve, 1000));  return {    lastResult: submission.reply(),    successMessage: 'Check your email for a reset link',  };}

CSS Variables

This component supports various CSS variables for theming. Here's a comprehensive list.

:root {  --forgot-password-font-family: var(--font-family-body);  --forgot-password-title-font-family: var(--font-family-heading);  --forgot-password-title: hsl(var(--foreground));  --forgot-password-subtitle: hsl(var(--contrast-500));}