Product Detail
Installation
Add the following Soul components
The product-detail component uses the streamable, accordion, price-label, rating, skeleton, breadcrumbs, product-gallery, button-radio-group, card-radio-group, checkbox, form-status, input, number-input, radio-group, select, swatch-radio-group and button components. Make sure you have added them to your project.
Install the following dependencies
npm install @conform-to/react @conform-to/zod nuqs zod
Copy and paste the following code into your project
sections/product-detail/index.tsx
import { ReactNode } from 'react';import { Stream, Streamable } from '@/vibes/soul/lib/streamable';import { Accordion, AccordionItem } from '@/vibes/soul/primitives/accordion';import { Price, PriceLabel } from '@/vibes/soul/primitives/price-label';import { Rating } from '@/vibes/soul/primitives/rating';import * as Skeleton from '@/vibes/soul/primitives/skeleton';import { type Breadcrumb, Breadcrumbs } from '@/vibes/soul/sections/breadcrumbs';import { ProductGallery } from '@/vibes/soul/sections/product-detail/product-gallery';import { ProductDetailForm, ProductDetailFormAction } from './product-detail-form';import { Field } from './schema';interface ProductDetailProduct { id: string; title: string; href: string; images: Streamable<Array<{ src: string; alt: string }>>; price?: Streamable<Price | null>; subtitle?: string; badge?: string; rating?: Streamable<number | null>; summary?: Streamable<string>; description?: Streamable<string | ReactNode | null>; accordions?: Streamable< Array<{ title: string; content: ReactNode; }> >;}export interface ProductDetailProps<F extends Field> { breadcrumbs?: Streamable<Breadcrumb[]>; product: Streamable<ProductDetailProduct | null>; action: ProductDetailFormAction<F>; fields: Streamable<F[]>; quantityLabel?: string; incrementLabel?: string; decrementLabel?: string; ctaLabel?: Streamable<string | null>; ctaDisabled?: Streamable<boolean | null>; prefetch?: boolean; thumbnailLabel?: string; additionalInformationTitle?: string;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { * --product-detail-border: hsl(var(--contrast-100)); * --product-detail-subtitle-font-family: var(--font-family-mono); * --product-detail-title-font-family: var(--font-family-heading); * --product-detail-primary-text: hsl(var(--foreground)); * --product-detail-secondary-text: hsl(var(--contrast-500)); * } * ``` */export function ProductDetail<F extends Field>({ product: streamableProduct, action, fields: streamableFields, breadcrumbs, quantityLabel, incrementLabel, decrementLabel, ctaLabel: streamableCtaLabel, ctaDisabled: streamableCtaDisabled, prefetch, thumbnailLabel, additionalInformationTitle = 'Additional information',}: ProductDetailProps<F>) { return ( <section className="@container"> <div className="group/product-detail mx-auto w-full max-w-screen-2xl px-4 py-10 @xl:px-6 @xl:py-14 @4xl:px-8 @4xl:py-20"> {breadcrumbs && ( <div className="group/breadcrumbs mb-6"> <Breadcrumbs breadcrumbs={breadcrumbs} /> </div> )} <Stream fallback={<ProductDetailSkeleton />} value={streamableProduct}> {(product) => product && ( <div className="grid grid-cols-1 items-stretch gap-x-8 gap-y-8 @2xl:grid-cols-2 @5xl:gap-x-12"> <div className="group/product-gallery hidden @2xl:block"> <Stream fallback={<ProductGallerySkeleton />} value={product.images}> {(images) => <ProductGallery images={images} />} </Stream> </div> {/* Product Details */} <div className="text-[var(--product-detail-primary-text,hsl(var(--foreground)))]"> {Boolean(product.subtitle) && ( <p className="font-[family-name:var(--product-detail-subtitle-font-family,var(--font-family-mono))] text-sm uppercase"> {product.subtitle} </p> )} <h1 className="mb-3 mt-2 font-[family-name:var(--product-detail-title-font-family,var(--font-family-heading))] text-2xl font-medium leading-none @xl:mb-4 @xl:text-3xl @4xl:text-4xl"> {product.title} </h1> <div className="group/product-rating"> <Stream fallback={<RatingSkeleton />} value={product.rating}> {(rating) => <Rating rating={rating ?? 0} />} </Stream> </div> <div className="group/product-price"> <Stream fallback={<PriceLabelSkeleton />} value={product.price}> {(price) => ( <PriceLabel className="my-3 text-xl @xl:text-2xl" price={price ?? ''} /> )} </Stream> </div> <div className="group/product-gallery mb-8 @2xl:hidden"> <Stream fallback={<ProductGallerySkeleton />} value={product.images}> {(images) => ( <ProductGallery images={images} thumbnailLabel={thumbnailLabel} /> )} </Stream> </div> <div className="group/product-summary"> <Stream fallback={<ProductSummarySkeleton />} value={product.summary}> {(summary) => Boolean(summary) && ( <p className="text-[var(--product-detail-secondary-text,hsl(var(--contrast-500)))]"> {summary} </p> ) } </Stream> </div> <div className="group/product-detail-form"> <Stream fallback={<ProductDetailFormSkeleton />} value={Streamable.all([ streamableFields, streamableCtaLabel, streamableCtaDisabled, ])} > {([fields, ctaLabel, ctaDisabled]) => ( <ProductDetailForm action={action} ctaDisabled={ctaDisabled ?? undefined} ctaLabel={ctaLabel ?? undefined} decrementLabel={decrementLabel} fields={fields} incrementLabel={incrementLabel} prefetch={prefetch} productId={product.id} quantityLabel={quantityLabel} /> )} </Stream> </div> <div className="group/product-description"> <Stream fallback={<ProductDescriptionSkeleton />} value={product.description}> {(description) => Boolean(description) && ( <div className="prose prose-sm max-w-none border-t border-[var(--product-detail-border,hsl(var(--contrast-100)))] py-8"> {description} </div> ) } </Stream> </div> <h2 className="sr-only">{additionalInformationTitle}</h2> <div className="group/product-accordion"> <Stream fallback={<ProductAccordionsSkeleton />} value={product.accordions}> {(accordions) => accordions && ( <Accordion className="border-t border-[var(--product-detail-border,hsl(var(--contrast-100)))] pt-4" type="multiple" > {accordions.map((accordion, index) => ( <AccordionItem key={index} title={accordion.title} value={index.toString()} > {accordion.content} </AccordionItem> ))} </Accordion> ) } </Stream> </div> </div> </div> ) } </Stream> </div> </section> );}function ProductGallerySkeleton() { return ( <Skeleton.Root className="group-has-[[data-pending]]/product-gallery:animate-pulse" pending> <div className="w-full overflow-hidden rounded-xl @xl:rounded-2xl"> <div className="flex"> <Skeleton.Box className="aspect-[4/5] h-full w-full shrink-0 grow-0 basis-full" /> </div> </div> <div className="mt-2 flex max-w-full gap-2 overflow-x-auto"> {Array.from({ length: 5 }).map((_, idx) => ( <Skeleton.Box className="h-12 w-12 shrink-0 rounded-lg @md:h-16 @md:w-16" key={idx} /> ))} </div> </Skeleton.Root> );}function PriceLabelSkeleton() { return <Skeleton.Box className="my-4 h-4 w-20 rounded-md" />;}function RatingSkeleton() { return ( <Skeleton.Root className="flex w-[136px] items-center gap-1 group-has-[[data-pending]]/product-rating:animate-pulse" pending > <Skeleton.Box className="h-4 w-[100px] rounded-md" /> <Skeleton.Box className="h-6 w-8 rounded-xl" /> </Skeleton.Root> );}function ProductSummarySkeleton() { return ( <Skeleton.Root className="flex w-full flex-col gap-3.5 pb-6 group-has-[[data-pending]]/product-summary:animate-pulse" pending > {Array.from({ length: 3 }).map((_, idx) => ( <Skeleton.Box className="h-2.5 w-full" key={idx} /> ))} </Skeleton.Root> );}function ProductDescriptionSkeleton() { return ( <Skeleton.Root className="flex w-full flex-col gap-3.5 pb-6 group-has-[[data-pending]]/product-description:animate-pulse" pending > {Array.from({ length: 2 }).map((_, idx) => ( <Skeleton.Box className="h-2.5 w-full" key={idx} /> ))} <Skeleton.Box className="h-2.5 w-3/4" /> </Skeleton.Root> );}function ProductDetailFormSkeleton() { return ( <Skeleton.Root className="flex flex-col gap-8 py-8 group-has-[[data-pending]]/product-detail-form:animate-pulse" pending > <div className="flex flex-col gap-5"> <Skeleton.Box className="h-2 w-10 rounded-md" /> <div className="flex gap-2"> {Array.from({ length: 3 }).map((_, idx) => ( <Skeleton.Box className="h-11 w-[72px] rounded-full" key={idx} /> ))} </div> </div> <div className="flex flex-col gap-5"> <Skeleton.Box className="h-3 w-16 rounded-md" /> <div className="flex gap-4"> {Array.from({ length: 5 }).map((_, idx) => ( <Skeleton.Box className="h-10 w-10 rounded-full" key={idx} /> ))} </div> </div> <div className="flex gap-2"> <Skeleton.Box className="h-12 w-[120px] rounded-lg" /> <Skeleton.Box className="h-12 w-[216px] rounded-full" /> </div> </Skeleton.Root> );}function ProductAccordionsSkeleton() { return ( <Skeleton.Root className="flex h-[600px] w-full flex-col gap-8 pt-4 group-has-[[data-pending]]/product-accordion:animate-pulse" pending > <div className="flex items-center justify-between"> <Skeleton.Box className="h-2 w-20 rounded-sm" /> <Skeleton.Box className="h-3 w-3 rounded-sm" /> </div> <div className="mb-1 flex flex-col gap-4"> <Skeleton.Box className="h-3 w-full rounded-sm" /> <Skeleton.Box className="h-3 w-full rounded-sm" /> <Skeleton.Box className="h-3 w-3/5 rounded-sm" /> </div> <div className="flex items-center justify-between"> <Skeleton.Box className="h-2 w-24 rounded-sm" /> <Skeleton.Box className="h-3 w-3 rounded-full" /> </div> <div className="flex items-center justify-between"> <Skeleton.Box className="h-2 w-20 rounded-sm" /> <Skeleton.Box className="h-3 w-3 rounded-full" /> </div> <div className="flex items-center justify-between"> <Skeleton.Box className="h-2 w-32 rounded-sm" /> <Skeleton.Box className="h-3 w-3 rounded-full" /> </div> </Skeleton.Root> );}export function ProductDetailSkeleton() { return ( <Skeleton.Root className="grid grid-cols-1 items-stretch gap-x-6 gap-y-8 group-has-[[data-pending]]/product-detail:animate-pulse @2xl:grid-cols-2 @5xl:gap-x-12" pending > <div className="hidden @2xl:block"> <ProductGallerySkeleton /> </div> <div> <Skeleton.Box className="mb-6 h-4 w-20 rounded-lg" /> <Skeleton.Box className="mb-6 h-6 w-72 rounded-lg" /> <RatingSkeleton /> <PriceLabelSkeleton /> <ProductSummarySkeleton /> <div className="mb-8 @2xl:hidden"> <ProductGallerySkeleton /> </div> <ProductDetailFormSkeleton /> </div> </Skeleton.Root> );}
sections/product-detail/product-detail-form.tsx
'use client';import { FieldMetadata, FormProvider, FormStateInput, getFormProps, SubmissionResult, useForm, useInputControl,} from '@conform-to/react';import { getZodConstraint, parseWithZod } from '@conform-to/zod';import { usePathname, useRouter } from 'next/navigation';import { createSerializer, parseAsString, useQueryStates } from 'nuqs';import { ReactNode, useActionState, useCallback, useEffect } from 'react';import { useFormStatus } from 'react-dom';import { z } from 'zod';import { ButtonRadioGroup } from '@/vibes/soul/form/button-radio-group';import { CardRadioGroup } from '@/vibes/soul/form/card-radio-group';import { Checkbox } from '@/vibes/soul/form/checkbox';import { FormStatus } from '@/vibes/soul/form/form-status';import { Input } from '@/vibes/soul/form/input';import { NumberInput } from '@/vibes/soul/form/number-input';import { RadioGroup } from '@/vibes/soul/form/radio-group';import { Select } from '@/vibes/soul/form/select';import { SwatchRadioGroup } from '@/vibes/soul/form/swatch-radio-group';import { Button } from '@/vibes/soul/primitives/button';import { toast } from '@/vibes/soul/primitives/toaster';import { Field, schema, SchemaRawShape } from './schema';type Action<S, P> = (state: Awaited<S>, payload: P) => S | Promise<S>;interface State<F extends Field> { fields: F[]; lastResult: SubmissionResult | null; successMessage?: ReactNode;}export type ProductDetailFormAction<F extends Field> = Action<State<F>, FormData>;export interface ProductDetailFormProps<F extends Field> { fields: F[]; action: ProductDetailFormAction<F>; productId: string; ctaLabel?: string; quantityLabel?: string; incrementLabel?: string; decrementLabel?: string; ctaDisabled?: boolean; prefetch?: boolean;}export function ProductDetailForm<F extends Field>({ action, fields, productId, ctaLabel = 'Add to cart', quantityLabel = 'Quantity', incrementLabel = 'Increase quantity', decrementLabel = 'Decrease quantity', ctaDisabled = false, prefetch = false,}: ProductDetailFormProps<F>) { const router = useRouter(); const pathname = usePathname(); const searchParams = fields.reduce<Record<string, typeof parseAsString>>((acc, field) => { return field.persist === true ? { ...acc, [field.name]: parseAsString } : acc; }, {}); const [params] = useQueryStates(searchParams, { shallow: false }); const onPrefetch = (fieldName: string, value: string) => { if (prefetch) { const serialize = createSerializer(searchParams); const newUrl = serialize(pathname, { ...params, [fieldName]: value }); router.prefetch(newUrl); } }; const defaultValue = fields.reduce<{ [Key in keyof SchemaRawShape]?: z.infer<SchemaRawShape[Key]>; }>( (acc, field) => ({ ...acc, [field.name]: params[field.name] ?? field.defaultValue ?? '', }), { quantity: 1 }, ); const [{ lastResult, successMessage }, formAction] = useActionState(action, { fields, lastResult: null, }); useEffect(() => { if (lastResult?.status === 'success') { toast.success(successMessage); } }, [lastResult, successMessage]); const [form, formFields] = useForm({ lastResult, constraint: getZodConstraint(schema(fields)), onValidate({ formData }) { return parseWithZod(formData, { schema: schema(fields) }); }, // @ts-expect-error: `defaultValue` types are conflicting with `onValidate`. defaultValue, shouldValidate: 'onSubmit', shouldRevalidate: 'onInput', }); const quantityControl = useInputControl(formFields.quantity); return ( <FormProvider context={form.context}> <FormStateInput /> <form {...getFormProps(form)} action={formAction} className="py-8"> <input name="id" type="hidden" value={productId} /> <div className="space-y-6"> {fields.map((field) => { return ( <FormField field={field} // eslint-disable-next-line @typescript-eslint/no-non-null-assertion formField={formFields[field.name]!} // eslint-disable-next-line @typescript-eslint/no-non-null-assertion key={formFields[field.name]!.id} onPrefetch={onPrefetch} /> ); })} {form.errors?.map((error, index) => ( <FormStatus className="pt-3" key={index} type="error"> {error} </FormStatus> ))} <div className="flex gap-x-3 pt-3"> <NumberInput aria-label={quantityLabel} decrementLabel={decrementLabel} incrementLabel={incrementLabel} min={1} name={formFields.quantity.name} onBlur={quantityControl.blur} onChange={(e) => quantityControl.change(e.currentTarget.value)} onFocus={quantityControl.focus} required value={quantityControl.value} /> <SubmitButton disabled={ctaDisabled}>{ctaLabel}</SubmitButton> </div> </div> </form> </FormProvider> );}function SubmitButton({ children, disabled }: { children: ReactNode; disabled?: boolean }) { const { pending } = useFormStatus(); return ( <Button className="w-auto @xl:w-56" disabled={disabled} loading={pending} size="medium" type="submit" > {children} </Button> );}function FormField({ field, formField, onPrefetch,}: { field: Field; formField: FieldMetadata<string | number | boolean | Date | undefined>; onPrefetch: (fieldName: string, value: string) => void;}) { const controls = useInputControl(formField); const [, setParams] = useQueryStates( field.persist === true ? { [field.name]: parseAsString.withOptions({ shallow: false }) } : {}, ); const handleChange = useCallback( (value: string) => { void setParams({ [field.name]: value }); controls.change(value); }, [setParams, field, controls], ); const handleOnOptionMouseEnter = (value: string) => { if (field.persist === true) { onPrefetch(field.name, value); } }; switch (field.type) { case 'number': return ( <NumberInput decrementLabel={field.decrementLabel} errors={formField.errors} incrementLabel={field.incrementLabel} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onChange={(e) => handleChange(e.currentTarget.value)} onFocus={controls.focus} required={formField.required} value={controls.value ?? ''} /> ); case 'text': return ( <Input errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onChange={(e) => handleChange(e.currentTarget.value)} onFocus={controls.focus} required={formField.required} value={controls.value ?? ''} /> ); case 'checkbox': return ( <Checkbox errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onCheckedChange={(value) => handleChange(String(value))} onFocus={controls.focus} required={formField.required} value={controls.value ?? 'false'} /> ); case 'select': return ( <Select errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onFocus={controls.focus} onOptionMouseEnter={handleOnOptionMouseEnter} onValueChange={handleChange} options={field.options} required={formField.required} value={controls.value ?? ''} /> ); case 'radio-group': return ( <RadioGroup errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onFocus={controls.focus} onOptionMouseEnter={handleOnOptionMouseEnter} onValueChange={handleChange} options={field.options} required={formField.required} value={controls.value ?? ''} /> ); case 'swatch-radio-group': return ( <SwatchRadioGroup errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onFocus={controls.focus} onOptionMouseEnter={handleOnOptionMouseEnter} onValueChange={handleChange} options={field.options} required={formField.required} value={controls.value ?? ''} /> ); case 'card-radio-group': return ( <CardRadioGroup errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onFocus={controls.focus} onOptionMouseEnter={handleOnOptionMouseEnter} onValueChange={handleChange} options={field.options} required={formField.required} value={controls.value ?? ''} /> ); case 'button-radio-group': return ( <ButtonRadioGroup errors={formField.errors} key={formField.id} label={field.label} name={formField.name} onBlur={controls.blur} onFocus={controls.focus} onOptionMouseEnter={handleOnOptionMouseEnter} onValueChange={handleChange} options={field.options} required={formField.required} value={controls.value ?? ''} /> ); }}
sections/product-detail/schema.ts
import { z } from 'zod';interface FormField { name: string; label?: string; errors?: string[]; required?: boolean; persist?: boolean;}type RadioField = { type: 'radio-group'; options: Array<{ label: string; value: string }>; defaultValue?: string;} & FormField;type SelectField = { type: 'select'; options: Array<{ label: string; value: string }>; defaultValue?: string;} & FormField;type CheckboxField = { type: 'checkbox'; defaultValue?: string;} & FormField;type NumberInputField = { type: 'number'; defaultValue?: string; min?: number; max?: number; incrementLabel?: string; decrementLabel?: string;} & FormField;type TextInputField = { type: 'text'; defaultValue?: string; pattern?: string;} & FormField;type TextAreaField = { type: 'textarea'; defaultValue?: string; pattern?: string;} & FormField;type DateField = { type: 'date'; defaultValue?: string; pattern?: string;} & FormField;type SwatchRadioFieldOption = | { type: 'color'; value: string; label: string; color: string; disabled?: boolean; } | { type: 'image'; value: string; label: string; image: { src: string; alt: string }; disabled?: boolean; };type SwatchRadioField = { type: 'swatch-radio-group'; defaultValue?: string; options: SwatchRadioFieldOption[];} & FormField;type CardRadioField = { type: 'card-radio-group'; defaultValue?: string; options: Array<{ value: string; label: string; image: { src: string; alt: string }; disabled?: boolean; }>;} & FormField;type ButtonRadioField = { type: 'button-radio-group'; defaultValue?: string; pattern?: string; options: Array<{ value: string; label: string; disabled?: boolean; }>;} & FormField;export type Field = | RadioField | CheckboxField | NumberInputField | TextInputField | TextAreaField | DateField | SwatchRadioField | CardRadioField | ButtonRadioField | SelectField;export interface SchemaRawShape { [key: string]: | z.ZodString | z.ZodOptional<z.ZodString> | z.ZodNumber | z.ZodOptional<z.ZodNumber>; id: z.ZodString; quantity: z.ZodNumber;}export function schema(fields: Field[]): z.ZodObject<SchemaRawShape> { const shape: SchemaRawShape = { id: z.string(), quantity: z.number().min(1), }; fields.forEach((field) => { let fieldSchema: z.ZodString | z.ZodNumber; switch (field.type) { case 'number': fieldSchema = z.number(); if (field.min != null) fieldSchema = fieldSchema.min(field.min); if (field.max != null) fieldSchema = fieldSchema.max(field.max); shape[field.name] = fieldSchema; break; default: fieldSchema = z.string(); shape[field.name] = fieldSchema; break; } if (field.required !== true) shape[field.name] = fieldSchema.optional(); }); return z.object(shape);}
Usage
import { ProductDetail } from '@/vibes/soul/sections/product-detail';function Usage() { return ( <ProductDetail action={action} breadcrumbs={breadcrumbs} fields={fieldsPromise} product={product} /> );}export const fields = [ { type: 'button-radio-group', label: 'Size', name: 'size', options: [ { value: 'sm', label: 'Small' }, { value: 'md', label: 'Medium' }, { value: 'lg', label: 'Large' }, ], required: true, }] export async function action( prevState: { fields: Field[]; lastResult: SubmissionResult | null }, payload: FormData,) { 'use server'; const submission = parseWithZod(payload, { schema: schema(fields) }); if (submission.status !== 'success') { return { fields: prevState.fields, lastResult: submission.reply(), }; } // Simulate add to cart await new Promise((resolve) => setTimeout(resolve, 1000)); return { fields: prevState.fields, lastResult: submission.reply({}), successMessage: 'Product(s) added to cart!', };}export const accordions = [ { title: 'What is your return policy?', content: 'We want you to be completely satisfied with your purchase. If you’re not happy with your bike pack, you can return it within 30 days of delivery. Please ensure the pack is in its original condition and packaging. For detailed return instructions, visit our Return Policy page or contact our customer support team.', },];export const product = { id: '1', title: 'Mini Bar Bag', price: '$60', image: { src: 'https://rstr.in/monogram/vibes/5IdIE27Cj9r', alt: 'A close-up of a bicycle handlebar with a brown handlebar bag.', }, images: [ { src: 'https://rstr.in/monogram/vibes/5IdIE27Cj9r', alt: 'A close-up of a bicycle handlebar with a brown handlebar bag.', }, ], href: '#', rating: 4.8, summary: 'A sleek, versatile bike bag designed to fit various bikes while holding essentials like snacks, phone, and tools. Multiple mounts ensure a secure, streamlined fit.', description: 'Svelte and functional, this is one bag that goes well with every bike. We made this smaller so it fits little bikes and still carries the essentials - snacks, wallet, phone, keys, a tube, and tools. With multiple mounting positions, the fit can be dialed for short head-tubed mountain bikes, long stemmed road bikes, and everything in-between. The slim top edge is designed to fit behind mountain bike cables and tuck up neatly under computers, lights, and other accessories.', accordions,};
API Reference
ProductDetailProps<F>
Prop | Type | Default |
---|---|---|
breadcrumbs | ||
product | ||
action | ProductDetailFormAction<F> | |
fields | Streamable <F[]> | |
quantityLabel | string | |
incrementLabel | string | |
decrementLabel | string | |
ctaLabel | Streamable <string | null> | |
ctaDisabled | Streamable <boolean | null> | |
prefetch | boolean | |
thumbnailLabel | string | |
additionalInformationTitle | string |
ProductDetailProduct
Prop | Type | Default |
---|---|---|
id | string | |
title | string | |
href | string | |
images | ||
price | ||
subtitle | string | |
badge | string | |
rating | Streamable <number | null> | |
summary | Streamable <string> | |
description | Streamable <string | ReactNode | null> | |
accordions |
Acorrdion
Prop | Type | Default |
---|---|---|
title* | string | |
content* | ReactNode |
Image
Prop | Type | Default |
---|---|---|
src* | string | |
alt* | string |
PriceRange
Prop | Type | Default |
---|---|---|
type | 'range' | |
minValue | string | |
maxValue | string |
PriceSale
Prop | Type | Default |
---|---|---|
type | 'sale' | |
previousValue | string | |
currentValue | string |
ProductDetailFormAction
interface State<F extends Field> { fields: F[]; lastResult: SubmissionResult | null; successMessage?: ReactNode;}export type ProductDetailFormAction<F extends Field> = Action<State<F>, 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:
export async function action( prevState: { fields: Field[]; lastResult: SubmissionResult | null }, payload: FormData,) { 'use server'; const submission = parseWithZod(payload, { schema: schema(fields) }); if (submission.status !== 'success') { return { fields: prevState.fields, lastResult: submission.reply(), }; } await new Promise((resolve) => setTimeout(resolve, 1000)); return { fields: prevState.fields, lastResult: submission.reply({}), successMessage: 'Product(s) added to cart!', };}
Breadcrumb
Prop | Type | Default |
---|---|---|
label* | string | |
href* | string |
CSS Variables
This component supports various CSS variables for theming. Here's a comprehensive list.
:root { --product-detail-border: hsl(var(--contrast-100)); --product-detail-subtitle-font-family: var(--font-family-mono); --product-detail-title-font-family: var(--font-family-heading); --product-detail-primary-text: hsl(var(--foreground)); --product-detail-secondary-text: hsl(var(--contrast-500));}