Featured Product Carousel
Installation
Add the following Soul components
The featured-product-carousel component uses the product-carousel, section-layout, animated-link and streamable components. Make sure you have added them to your project.
Copy and paste the following code into your project
sections/featured-product-carousel/index.tsx
import { Streamable } from '@/vibes/soul/lib/streamable';import { AnimatedLink } from '@/vibes/soul/primitives/animated-link';import { CarouselProduct, ProductCarousel } from '@/vibes/soul/sections/product-carousel';import { SectionLayout } from '@/vibes/soul/sections/section-layout';interface Link { label: string; href: string;}export interface FeaturedProductCarouselProps { title: string; description?: string; cta?: Link; products: Streamable<CarouselProduct[]>; emptyStateTitle?: Streamable<string>; emptyStateSubtitle?: Streamable<string>; placeholderCount?: number; scrollbarLabel?: string; previousLabel?: string; nextLabel?: string; hideOverflow?: boolean;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { * --featured-product-carousel-font-family: var(--font-family-body); * --featured-product-carousel-title-font-family: var(--font-family-heading); * --featured-product-carousel-title: hsl(var(--foreground)); * --featured-product-carousel-description: hsl(var(--contrast-500)); * } * ``` */export function FeaturedProductCarousel({ title, description, cta, products, emptyStateTitle, emptyStateSubtitle, placeholderCount, scrollbarLabel, previousLabel, nextLabel, hideOverflow = false,}: FeaturedProductCarouselProps) { return ( <SectionLayout containerSize="2xl"> <div className="mb-6 flex w-full flex-row flex-wrap items-end justify-between gap-x-8 gap-y-6 @4xl:mb-8"> <header className="font-[family-name:var(--featured-product-carousel-font-family,var(--font-family-body))]"> <h2 className="font-[family-name:var(--featured-product-carousel-title-font-family,var(--font-family-heading))] text-2xl leading-none text-[var(--featured-product-carousel-title,hsl(var(--foreground)))] @xl:text-3xl @4xl:text-4xl"> {title} </h2> {description != null && description !== '' && ( <p className="mt-3 max-w-xl leading-relaxed text-[var(--featured-product-carousel-description,hsl(var(--contrast-500)))]"> {description} </p> )} </header> {cta != null && cta.href !== '' && cta.label !== '' && ( <AnimatedLink className="mr-3" href={cta.href}> {cta.label} </AnimatedLink> )} </div> <div className="group/product-carousel"> <ProductCarousel emptyStateSubtitle={emptyStateSubtitle} emptyStateTitle={emptyStateTitle} hideOverflow={hideOverflow} nextLabel={nextLabel} placeholderCount={placeholderCount} previousLabel={previousLabel} products={products} scrollbarLabel={scrollbarLabel} /> </div> </SectionLayout> );}
Usage
import { FeaturedProductCarousel } from '@/vibes/soul/sections/featured-product-carousel';function Usage() { return ( <FeaturedProductCarousel cta={{ label: 'Shop Now', href: '#' }} description="Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore." products={products} title="Our plants" /> );}const products = [ { id: '1', title: 'Philodendron Imperial Red', subtitle: 'Indoor Plant', badge: 'Popular', price: '$44.95', image: { src: 'https://rstr.in/monogram/vibes/-kv08IvX08j', alt: 'Philodendron Imperial Red', }, href: '#', rating: 4, }]
API Reference
FeaturedProductCarouselProps
Prop | Type | Default |
---|---|---|
className | string | |
title | string | |
description | string | |
cta | Link | |
products* | ||
emptyStateTitle | Streamable <string> | |
emptyStateSubtitle | Streamable <string> | |
placeholderCount | number | |
scrollbarLabel | string | |
previousLabel | string | |
nextLabel | string | |
hideOverflow | boolean |
CarouselProduct
Prop | Type | Default |
---|---|---|
id* | string | |
title* | string | |
href* | string | |
image | { src: string; alt: string } | |
price | string | PriceRange | PriceSale | |
subtitle | string | |
badge | string | |
rating | number |
PriceRange
Prop | Type | Default |
---|---|---|
type* | 'range' | |
minValue* | string | |
maxValue* | string |
PriceSale
Prop | Type | Default |
---|---|---|
type* | 'sale' | |
previousValue* | string | |
currentValue* | string |
CSS Variables
This component supports various CSS variables for theming. Here's a comprehensive list.
:root { --featured-product-carousel-font-family: var(--font-family-body); --featured-product-carousel-title-font-family: var(--font-family-heading); --featured-product-carousel-title: hsl(var(--foreground)); --featured-product-carousel-description: hsl(var(--contrast-500));}