CrematoriumStep: rewrite to match live site variants

Simplified from over-engineered multi-card selection to two clean
variants based on funeral type:

- Service & Cremation: compact card + witness Yes/No (ToggleButtonGroup)
- Cremation Only: compact card + badge + "Included in Package" notice

Removed: multi-card grid, priority dropdown, special instructions,
crematoriums array prop. Crematorium is always pre-selected by provider.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-03-30 21:51:03 +11:00
parent a8354cc8dd
commit ab9c8bfa6f
3 changed files with 201 additions and 311 deletions

View File

@@ -83,7 +83,7 @@ duplicates) and MUST update it after completing one.
| VenueStep | done | WizardLayout (centered-form) + VenueCard + AddOnOption + Collapse + Chip + TextField + Divider + Button | Wizard step 7a — venue browsing. Click-to-navigate card grid with search/filters. Leads to VenueDetailStep. | | VenueStep | done | WizardLayout (centered-form) + VenueCard + AddOnOption + Collapse + Chip + TextField + Divider + Button | Wizard step 7a — venue browsing. Click-to-navigate card grid with search/filters. Leads to VenueDetailStep. |
| VenueDetailStep | done | WizardLayout (detail-toggles) + ImageGallery + Card + Chip + Typography + Button + Divider | Wizard step 7b — venue detail. Two-panel: gallery/description/features/location (left), name/meta/price/CTA/religions (right). Informational service preview. | | VenueDetailStep | done | WizardLayout (detail-toggles) + ImageGallery + Card + Chip + Typography + Button + Divider | Wizard step 7b — venue detail. Two-panel: gallery/description/features/location (left), name/meta/price/CTA/religions (right). Informational service preview. |
| VenueServicesStep | done | WizardLayout (centered-form) + AddOnOption + Card + Typography + Button + Divider | Wizard step 7c — venue services. Compact venue card, availability notices, AddOnOption toggles with "View more" for long descriptions. Follows VenueDetailStep. | | VenueServicesStep | done | WizardLayout (centered-form) + AddOnOption + Card + Typography + Button + Divider | Wizard step 7c — venue services. Compact venue card, availability notices, AddOnOption toggles with "View more" for long descriptions. Follows VenueDetailStep. |
| CrematoriumStep | done | WizardLayout (centered-form) + Card + RadioGroup + Collapse + TextField + Divider + Button | Wizard step 8 — crematorium. Single confirmation card or multi-card grid. Witness question personalised with deceased name. Special instructions textarea. | | CrematoriumStep | done | WizardLayout (centered-form) + Card + Badge + ToggleButtonGroup + Typography + Button + Divider | Wizard step 8 — crematorium. Two variants: Service & Cremation (compact card + witness Yes/No toggle), Cremation Only (compact card + "Cremation Only" badge + "Included in Package" notice). Single pre-selected crematorium, no multi-select. |
| CemeteryStep | done | WizardLayout (centered-form) + Card + RadioGroup + Collapse + Divider + Button | Wizard step 9 — cemetery. Triple progressive disclosure (have plot? → choose? → grid). Dependent field resets. | | CemeteryStep | done | WizardLayout (centered-form) + Card + RadioGroup + Collapse + Divider + Button | Wizard step 9 — cemetery. Triple progressive disclosure (have plot? → choose? → grid). Dependent field resets. |
| CoffinsStep | done | WizardLayout (centered-form) + Card + Badge + TextField + MenuItem + Pagination + Divider + Button | Wizard step 10 — coffin selection. 3-col card grid with category/price filters. "Most Popular" badge. Pagination. | | CoffinsStep | done | WizardLayout (centered-form) + Card + Badge + TextField + MenuItem + Pagination + Divider + Button | Wizard step 10 — coffin selection. 3-col card grid with category/price filters. "Most Popular" badge. Pagination. |
| CoffinDetailsStep | done | WizardLayout (centered-form) + Paper + RadioGroup + Divider + Button | Wizard step 11 — coffin customisation. Profile (image + specs) + 3 option sections (handles, lining, nameplate). Branded selected state. | | CoffinDetailsStep | done | WizardLayout (centered-form) + Paper + RadioGroup + Divider + Button | Wizard step 11 — coffin customisation. Profile (image + specs) + 3 option sections (handles, lining, nameplate). Branded selected state. |

View File

@@ -1,7 +1,7 @@
import { useState } from 'react'; import { useState } from 'react';
import type { Meta, StoryObj } from '@storybook/react'; import type { Meta, StoryObj } from '@storybook/react';
import { CrematoriumStep } from './CrematoriumStep'; import { CrematoriumStep } from './CrematoriumStep';
import type { CrematoriumStepValues, CrematoriumStepErrors, Crematorium } from './CrematoriumStep'; import type { CrematoriumStepValues, Crematorium } from './CrematoriumStep';
import { Navigation } from '../../organisms/Navigation'; import { Navigation } from '../../organisms/Navigation';
import Box from '@mui/material/Box'; import Box from '@mui/material/Box';
@@ -34,46 +34,22 @@ const nav = (
/> />
); );
const singleCrematorium: Crematorium[] = [ const sampleCrematorium: Crematorium = {
{
id: 'warrill-park',
name: 'Warrill Park Crematorium', name: 'Warrill Park Crematorium',
location: 'Ipswich', imageUrl: 'https://images.unsplash.com/photo-1585320806297-9794b3e4eeae?w=400&h=300&fit=crop',
distance: '15 min from venue', address: 'Anderson Day Drive, Willowbank, QLD, 4306',
price: 850, distance: '152.62km from Killick Family Funerals Chapel Kingaroy',
}, };
];
const multipleCrematoriums: Crematorium[] = [ const sydneyCrematorium: Crematorium = {
{ name: 'Macquarie Park Cemetery & Crematorium',
id: 'warrill-park', imageUrl: 'https://images.unsplash.com/photo-1506905925346-21bda4d32df4?w=400&h=300&fit=crop',
name: 'Warrill Park Crematorium', address: 'Plassey Rd, Macquarie Park, NSW, 2113',
location: 'Ipswich', distance: '3.54km from Gladesville, Sydney',
distance: '15 min from venue', };
price: 850,
},
{
id: 'mt-gravatt',
name: 'Mt Gravatt Crematorium',
location: 'Mt Gravatt',
distance: '25 min from venue',
price: 920,
},
{
id: 'pinnaroo',
name: 'Pinnaroo Valley Memorial Park',
location: 'Padstow',
distance: '35 min from venue',
price: 780,
},
];
const defaultValues: CrematoriumStepValues = { const defaultValues: CrematoriumStepValues = {
selectedCrematoriumId: null, attend: null,
attend: 'no',
priority: '',
hasInstructions: 'no',
specialInstructions: '',
}; };
// ─── Meta ──────────────────────────────────────────────────────────────────── // ─── Meta ────────────────────────────────────────────────────────────────────
@@ -90,23 +66,20 @@ const meta: Meta<typeof CrematoriumStep> = {
export default meta; export default meta;
type Story = StoryObj<typeof CrematoriumStep>; type Story = StoryObj<typeof CrematoriumStep>;
// ─── Single crematorium (most common) ─────────────────────────────────────── // ─── Service & Cremation (default) ─────────────────────────────────────────
/** Single pre-selected crematorium — confirmation pattern */ /** Service & Cremation — witness question with personalised name */
export const Default: Story = { export const Default: Story = {
render: () => { render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({ const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues });
...defaultValues,
selectedCrematoriumId: 'warrill-park',
});
return ( return (
<CrematoriumStep <CrematoriumStep
crematorium={sampleCrematorium}
values={values} values={values}
onChange={setValues} onChange={setValues}
onContinue={() => alert('Continue')} onContinue={() => alert('Continue')}
onBack={() => alert('Back')} onBack={() => alert('Back')}
onSaveAndExit={() => alert('Save')} onSaveAndExit={() => alert('Save')}
crematoriums={singleCrematorium}
deceasedName="Margaret" deceasedName="Margaret"
navigation={nav} navigation={nav}
/> />
@@ -114,82 +87,40 @@ export const Default: Story = {
}, },
}; };
// ─── Multiple crematoriums ────────────────────────────────────────────────── // ─── Cremation Only ────────────────────────────────────────────────────────
/** Multiple options — card grid with selection */ /** Cremation Only — no witness question, "included in package" notice */
export const MultipleCrematoriums: Story = { export const CremationOnly: Story = {
render: () => { render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues }); const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues });
const [errors, setErrors] = useState<CrematoriumStepErrors>({});
const handleContinue = () => {
if (!values.selectedCrematoriumId) {
setErrors({ selectedCrematoriumId: 'Please confirm the crematorium.' });
return;
}
alert(`Selected: ${values.selectedCrematoriumId}`);
};
return (
<CrematoriumStep
values={values}
onChange={(v) => {
setValues(v);
setErrors({});
}}
onContinue={handleContinue}
onBack={() => alert('Back')}
errors={errors}
crematoriums={multipleCrematoriums}
deceasedName="Margaret"
navigation={nav}
/>
);
},
};
// ─── With special instructions ──────────────────────────────────────────────
/** Special instructions textarea revealed */
export const WithInstructions: Story = {
render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({
...defaultValues,
selectedCrematoriumId: 'warrill-park',
attend: 'yes',
hasInstructions: 'yes',
specialInstructions: 'Please ensure the family has 10 minutes for a private farewell.',
});
return ( return (
<CrematoriumStep <CrematoriumStep
crematorium={sydneyCrematorium}
values={values} values={values}
onChange={setValues} onChange={setValues}
onContinue={() => alert('Continue')} onContinue={() => alert('Continue')}
onBack={() => alert('Back')} onBack={() => alert('Back')}
crematoriums={singleCrematorium} onSaveAndExit={() => alert('Save')}
deceasedName="Margaret" isCremationOnly
navigation={nav} navigation={nav}
/> />
); );
}, },
}; };
// ─── Pre-planning ────────────────────────────────────────────────────────── // ─── Pre-planning ──────────────────────────────────────────────────────────
/** Pre-planning variant — softer helper text */ /** Pre-planning variant — softer copy */
export const PrePlanning: Story = { export const PrePlanning: Story = {
render: () => { render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({ const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues });
...defaultValues,
selectedCrematoriumId: 'warrill-park',
});
return ( return (
<CrematoriumStep <CrematoriumStep
crematorium={sampleCrematorium}
values={values} values={values}
onChange={setValues} onChange={setValues}
onContinue={() => alert('Continue')} onContinue={() => alert('Continue')}
onBack={() => alert('Back')} onBack={() => alert('Back')}
crematoriums={singleCrematorium}
isPrePlanning isPrePlanning
navigation={nav} navigation={nav}
/> />
@@ -197,19 +128,46 @@ export const PrePlanning: Story = {
}, },
}; };
// ─── Validation error ─────────────────────────────────────────────────────── // ─── No image ──────────────────────────────────────────────────────────────
/** No crematorium selected with error */ /** Crematorium without a photo */
export const WithError: Story = { export const NoImage: Story = {
render: () => { render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues }); const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues });
return ( return (
<CrematoriumStep <CrematoriumStep
crematorium={{
name: 'Northern Suburbs Crematorium',
address: '199 Delhi Rd, North Ryde, NSW, 2113',
distance: '8km from venue',
}}
values={values} values={values}
onChange={setValues} onChange={setValues}
onContinue={() => {}} onContinue={() => alert('Continue')}
errors={{ selectedCrematoriumId: 'Please confirm the crematorium.' }} onBack={() => alert('Back')}
crematoriums={multipleCrematoriums} onSaveAndExit={() => alert('Save')}
deceasedName="Robert"
navigation={nav}
/>
);
},
};
// ─── With error ────────────────────────────────────────────────────────────
/** Validation error on witness question */
export const WithError: Story = {
render: () => {
const [values, setValues] = useState<CrematoriumStepValues>({ ...defaultValues });
return (
<CrematoriumStep
crematorium={sampleCrematorium}
values={values}
onChange={setValues}
onContinue={() => {}}
onBack={() => alert('Back')}
errors={{ attend: 'Please let us know if anyone will follow the hearse.' }}
deceasedName="Margaret"
navigation={nav} navigation={nav}
/> />
); );

View File

@@ -1,58 +1,47 @@
import React from 'react'; import React from 'react';
import Box from '@mui/material/Box'; import Box from '@mui/material/Box';
import TextField from '@mui/material/TextField'; import LocationOnOutlinedIcon from '@mui/icons-material/LocationOnOutlined';
import MenuItem from '@mui/material/MenuItem'; import DirectionsCarOutlinedIcon from '@mui/icons-material/DirectionsCarOutlined';
import FormControl from '@mui/material/FormControl'; import InfoOutlinedIcon from '@mui/icons-material/InfoOutlined';
import FormLabel from '@mui/material/FormLabel';
import FormControlLabel from '@mui/material/FormControlLabel';
import RadioGroup from '@mui/material/RadioGroup';
import Radio from '@mui/material/Radio';
import type { SxProps, Theme } from '@mui/material/styles'; import type { SxProps, Theme } from '@mui/material/styles';
import { WizardLayout } from '../../templates/WizardLayout'; import { WizardLayout } from '../../templates/WizardLayout';
import { Card } from '../../atoms/Card'; import { Card } from '../../atoms/Card';
import { Collapse } from '../../atoms/Collapse'; import { Badge } from '../../atoms/Badge';
import { ToggleButtonGroup } from '../../atoms/ToggleButtonGroup';
import { Typography } from '../../atoms/Typography'; import { Typography } from '../../atoms/Typography';
import { Button } from '../../atoms/Button'; import { Button } from '../../atoms/Button';
import { Divider } from '../../atoms/Divider'; import { Divider } from '../../atoms/Divider';
// ─── Types ─────────────────────────────────────────────────────────────────── // ─── Types ───────────────────────────────────────────────────────────────────
/** A crematorium available for selection */ /** The pre-selected crematorium */
export interface Crematorium { export interface Crematorium {
id: string;
name: string; name: string;
location: string; imageUrl?: string;
address: string;
distance?: string; distance?: string;
price?: number;
} }
/** Form values for the crematorium step */ /** Form values for the crematorium step */
export interface CrematoriumStepValues { export interface CrematoriumStepValues {
/** Selected crematorium ID */ /** Will anyone follow the hearse? (service & cremation only) */
selectedCrematoriumId: string | null; attend: 'yes' | 'no' | null;
/** Will anyone follow the hearse? */
attend: 'yes' | 'no';
/** Cremation timing preference */
priority: string;
/** Has special instructions */
hasInstructions: 'yes' | 'no';
/** Special instructions text */
specialInstructions: string;
} }
/** Field-level error messages */ /** Field-level error messages */
export interface CrematoriumStepErrors { export interface CrematoriumStepErrors {
selectedCrematoriumId?: string;
attend?: string; attend?: string;
} }
/** Props for the CrematoriumStep page component */ /** Props for the CrematoriumStep page component */
export interface CrematoriumStepProps { export interface CrematoriumStepProps {
/** The pre-selected crematorium */
crematorium: Crematorium;
/** Current form values */ /** Current form values */
values: CrematoriumStepValues; values: CrematoriumStepValues;
/** Callback when any field value changes */ /** Callback when any field value changes */
onChange: (values: CrematoriumStepValues) => void; onChange: (values: CrematoriumStepValues) => void;
/** Callback when the Continue button is clicked */ /** Callback when Continue is clicked */
onContinue: () => void; onContinue: () => void;
/** Callback for back navigation */ /** Callback for back navigation */
onBack?: () => void; onBack?: () => void;
@@ -60,17 +49,15 @@ export interface CrematoriumStepProps {
onSaveAndExit?: () => void; onSaveAndExit?: () => void;
/** Field-level validation errors */ /** Field-level validation errors */
errors?: CrematoriumStepErrors; errors?: CrematoriumStepErrors;
/** Whether the Continue button is in a loading state */ /** Whether Continue is in a loading state */
loading?: boolean; loading?: boolean;
/** Available crematoriums */ /** Whether this is a cremation-only flow (no service) */
crematoriums: Crematorium[]; isCremationOnly?: boolean;
/** Priority/timing options (provider-specific) */ /** Deceased first name — personalises witness question */
priorityOptions?: Array<{ value: string; label: string }>;
/** Deceased first name — used to personalise copy */
deceasedName?: string; deceasedName?: string;
/** Whether this is a pre-planning flow */ /** Whether this is a pre-planning flow */
isPrePlanning?: boolean; isPrePlanning?: boolean;
/** Navigation bar — passed through to WizardLayout */ /** Navigation bar */
navigation?: React.ReactNode; navigation?: React.ReactNode;
/** Progress stepper */ /** Progress stepper */
progressStepper?: React.ReactNode; progressStepper?: React.ReactNode;
@@ -78,7 +65,7 @@ export interface CrematoriumStepProps {
runningTotal?: React.ReactNode; runningTotal?: React.ReactNode;
/** Hide the help bar */ /** Hide the help bar */
hideHelpBar?: boolean; hideHelpBar?: boolean;
/** MUI sx prop for the root */ /** MUI sx prop */
sx?: SxProps<Theme>; sx?: SxProps<Theme>;
} }
@@ -87,19 +74,21 @@ export interface CrematoriumStepProps {
/** /**
* Step 8 — Crematorium for the FA arrangement wizard. * Step 8 — Crematorium for the FA arrangement wizard.
* *
* Select a crematorium and cremation witness/attendance preferences. * Confirmation page for the provider's pre-selected crematorium.
* Only shown for cremation-type funerals. * Two variants based on funeral type:
* *
* Often a single pre-selected crematorium (provider's default). When * **Service & Cremation:** Crematorium card + witness question
* multiple options exist, shows a card grid with radiogroup pattern. * ("Will anyone follow the hearse?") with personalised helper text.
* *
* Personalises witness question with deceased name when available. * **Cremation Only:** Crematorium card + "Cremation Only" badge +
* "Included in Package" notice. No witness question (no procession).
* *
* Pure presentation component — props in, callbacks out. * Pure presentation component — props in, callbacks out.
* *
* Spec: documentation/steps/steps/08_crematorium.yaml * Spec: documentation/steps/steps/08_crematorium.yaml
*/ */
export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({ export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({
crematorium,
values, values,
onChange, onChange,
onContinue, onContinue,
@@ -107,8 +96,7 @@ export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({
onSaveAndExit, onSaveAndExit,
errors, errors,
loading = false, loading = false,
crematoriums, isCremationOnly = false,
priorityOptions = [],
deceasedName, deceasedName,
isPrePlanning = false, isPrePlanning = false,
navigation, navigation,
@@ -117,15 +105,6 @@ export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({
hideHelpBar, hideHelpBar,
sx, sx,
}) => { }) => {
const isSingle = crematoriums.length === 1;
const handleFieldChange = <K extends keyof CrematoriumStepValues>(
field: K,
value: CrematoriumStepValues[K],
) => {
onChange({ ...values, [field]: value });
};
const witnessHelper = deceasedName const witnessHelper = deceasedName
? `Please indicate if you wish to follow the hearse in your own vehicles to escort ${deceasedName} to the crematorium.` ? `Please indicate if you wish to follow the hearse in your own vehicles to escort ${deceasedName} to the crematorium.`
: isPrePlanning : isPrePlanning
@@ -152,6 +131,8 @@ export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({
<Typography variant="body1" color="text.secondary" sx={{ mb: 5 }}> <Typography variant="body1" color="text.secondary" sx={{ mb: 5 }}>
{isPrePlanning {isPrePlanning
? 'Review the crematorium details. You can update this later.' ? 'Review the crematorium details. You can update this later.'
: isCremationOnly
? 'The crematorium included with your selected package.'
: 'Confirm the crematorium and let us know about any preferences.'} : 'Confirm the crematorium and let us know about any preferences.'}
</Typography> </Typography>
@@ -164,173 +145,124 @@ export const CrematoriumStep: React.FC<CrematoriumStepProps> = ({
if (!loading) onContinue(); if (!loading) onContinue();
}} }}
> >
{/* ─── Crematorium selection ─── */} {/* ─── Compact crematorium card ─── */}
<Box sx={{ mb: 4 }}>
{isSingle ? (
// Single crematorium — presented as confirmation
<Card <Card
selected={values.selectedCrematoriumId === crematoriums[0].id} variant="outlined"
interactive padding="none"
onClick={() => handleFieldChange('selectedCrematoriumId', crematoriums[0].id)}
role="radio"
aria-checked={values.selectedCrematoriumId === crematoriums[0].id}
tabIndex={0}
sx={{ p: 3 }}
>
<Typography variant="h5">{crematoriums[0].name}</Typography>
<Typography variant="body2" color="text.secondary">
{crematoriums[0].location}
{crematoriums[0].distance && ` \u2022 ${crematoriums[0].distance}`}
</Typography>
{crematoriums[0].price != null && (
<Typography variant="h6" color="primary" sx={{ mt: 1 }}>
${crematoriums[0].price.toLocaleString('en-AU')}
</Typography>
)}
</Card>
) : (
// Multiple crematoriums — radiogroup grid
<Box role="radiogroup" aria-label="Available crematoriums">
<Box
sx={{ sx={{
display: 'grid', display: 'flex',
gridTemplateColumns: { xs: '1fr', sm: 'repeat(2, 1fr)' }, overflow: 'hidden',
gap: 2, minHeight: 100,
mb: 4,
}} }}
> >
{crematoriums.map((crem, index) => ( {crematorium.imageUrl && (
<Card <Box
key={crem.id} role="img"
interactive aria-label={`${crematorium.name} photo`}
selected={crem.id === values.selectedCrematoriumId} sx={{
onClick={() => handleFieldChange('selectedCrematoriumId', crem.id)} width: { xs: 120, sm: 160 },
role="radio" flexShrink: 0,
aria-checked={crem.id === values.selectedCrematoriumId} backgroundImage: `url(${crematorium.imageUrl})`,
tabIndex={ backgroundSize: 'cover',
values.selectedCrematoriumId === null backgroundPosition: 'center',
? index === 0 }}
? 0 />
: -1 )}
: crem.id === values.selectedCrematoriumId <Box
? 0 sx={{
: -1 flex: 1,
} display: 'flex',
sx={{ p: 3 }} flexDirection: 'column',
justifyContent: 'center',
gap: 0.5,
p: 2,
minWidth: 0,
}}
> >
<Typography variant="h5">{crem.name}</Typography> <Typography variant="h6" component="span" maxLines={2}>
{crematorium.name}
</Typography>
<Box sx={{ display: 'flex', alignItems: 'center', gap: 0.5 }}>
<LocationOnOutlinedIcon sx={{ fontSize: 14, color: 'text.secondary' }} aria-hidden />
<Typography variant="body2" color="text.secondary"> <Typography variant="body2" color="text.secondary">
{crem.location} {crematorium.address}
{crem.distance && ` \u2022 ${crem.distance}`}
</Typography> </Typography>
{crem.price != null && ( </Box>
<Typography variant="h6" color="primary" sx={{ mt: 1 }}> {crematorium.distance && (
${crem.price.toLocaleString('en-AU')} <Box sx={{ display: 'flex', alignItems: 'center', gap: 0.5 }}>
<DirectionsCarOutlinedIcon
sx={{ fontSize: 14, color: 'text.secondary' }}
aria-hidden
/>
<Typography variant="body2" color="text.secondary">
{crematorium.distance}
</Typography> </Typography>
</Box>
)} )}
</Box>
</Card> </Card>
))}
</Box>
</Box>
)}
{errors?.selectedCrematoriumId && ( {isCremationOnly ? (
<Typography <>
variant="body2" {/* ─── Cremation Only: badge + included notice ─── */}
sx={{ mt: 1, color: 'var(--fa-color-text-brand)' }} <Badge variant="soft" color="success" sx={{ mb: 3 }}>
role="alert" Cremation Only
</Badge>
<Card
variant="outlined"
padding="compact"
sx={{
bgcolor: 'var(--fa-color-surface-subtle)',
borderColor: 'var(--fa-color-border-default)',
}}
> >
{errors.selectedCrematoriumId} <Box sx={{ display: 'flex', gap: 1.5, alignItems: 'flex-start' }}>
<InfoOutlinedIcon
sx={{ fontSize: 20, color: 'text.secondary', mt: 0.25, flexShrink: 0 }}
aria-hidden
/>
<Box>
<Typography variant="label" sx={{ display: 'block', mb: 0.5 }}>
Crematorium included in package
</Typography>
<Typography variant="body2" color="text.secondary">
The selected crematorium is included in your funeral package. If you have any
questions or would prefer a different option, please speak with your funeral
director.
</Typography> </Typography>
)}
</Box> </Box>
</Box>
<Divider sx={{ my: 4 }} /> </Card>
</>
{/* ─── Witness / attendance question ─── */} ) : (
<FormControl component="fieldset" sx={{ mb: 4, display: 'block' }}> <>
<FormLabel component="legend" sx={{ mb: 0.5 }}> {/* ─── Service & Cremation: witness question ─── */}
<Typography variant="h5" component="h2" sx={{ mb: 0.5 }}>
Will anyone be following the hearse to the crematorium? Will anyone be following the hearse to the crematorium?
</FormLabel> </Typography>
<Typography variant="body2" color="text.secondary" sx={{ mb: 1.5 }}> <Typography variant="body2" color="text.secondary" sx={{ mb: 3 }}>
{witnessHelper} {witnessHelper}
</Typography> </Typography>
<RadioGroup
value={values.attend}
onChange={(e) =>
handleFieldChange('attend', e.target.value as CrematoriumStepValues['attend'])
}
>
<FormControlLabel value="yes" control={<Radio />} label="Yes" />
<FormControlLabel value="no" control={<Radio />} label="No" />
</RadioGroup>
{errors?.attend && (
<Typography
variant="body2"
sx={{ mt: 0.5, color: 'var(--fa-color-text-brand)' }}
role="alert"
>
{errors.attend}
</Typography>
)}
</FormControl>
{/* ─── Priority / timing preference (optional, provider-specific) ─── */} <ToggleButtonGroup
{priorityOptions.length > 0 && ( label=""
<> options={[
<Divider sx={{ my: 4 }} /> { value: 'yes', label: 'Yes' },
<TextField { value: 'no', label: 'No' },
select ]}
label="Cremation timing preference" value={values.attend}
value={values.priority} onChange={(v) =>
onChange={(e) => handleFieldChange('priority', e.target.value)} onChange({ ...values, attend: v as CrematoriumStepValues['attend'] })
placeholder="Select timing preference (optional)" }
error={!!errors?.attend}
helperText={errors?.attend}
fullWidth fullWidth
sx={{ mb: 3 }} />
>
{priorityOptions.map((opt) => (
<MenuItem key={opt.value} value={opt.value}>
{opt.label}
</MenuItem>
))}
</TextField>
</> </>
)} )}
<Divider sx={{ my: 4 }} />
{/* ─── Special instructions ─── */}
<FormControl component="fieldset" sx={{ mb: 3, display: 'block' }}>
<FormLabel component="legend" sx={{ mb: 1 }}>
Do you have any special requests for the cremation?
</FormLabel>
<RadioGroup
value={values.hasInstructions}
onChange={(e) =>
handleFieldChange(
'hasInstructions',
e.target.value as CrematoriumStepValues['hasInstructions'],
)
}
>
<FormControlLabel value="no" control={<Radio />} label="No" />
<FormControlLabel value="yes" control={<Radio />} label="Yes" />
</RadioGroup>
</FormControl>
<Collapse in={values.hasInstructions === 'yes'}>
<TextField
label="Special requests"
value={values.specialInstructions}
onChange={(e) => handleFieldChange('specialInstructions', e.target.value)}
placeholder="Enter any special requests or instructions..."
multiline
minRows={3}
maxRows={8}
fullWidth
sx={{ mb: 3 }}
/>
</Collapse>
<Divider sx={{ my: 3 }} /> <Divider sx={{ my: 3 }} />
{/* CTAs */} {/* CTAs */}