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

PropTypeDefault
className
string
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

CarouselProduct

PropTypeDefault
id*
string
title*
string
href*
string
image
{ src: string; alt: string }
price
string | PriceRange | PriceSale
subtitle
string
badge
string
rating
number

PriceRange

PropTypeDefault
type*
'range'
minValue*
string
maxValue*
string

PriceSale

PropTypeDefault
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));}