Gallery 13



<section id="relume">
<div class="px-[5%] py-16 md:py-24 lg:py-28">
<div class="container text-center">
<h2 class="rb-5 mb-5 text-5xl font-bold md:mb-6 md:text-7xl lg:text-8xl">Image Gallery</h2>
<p class="md:text-md">Lorem ipsum dolor sit amet, consectetur adipiscing elit.</p>
</div>
</div>
<div class="relative overflow-hidden" role="region" aria-roledescription="carousel">
<div>
<div class="flex ml-0">
<div
role="group"
aria-roledescription="slide"
class="min-w-0 shrink-0 grow-0 basis-full relative h-dvh pl-0"
>
<img
src="https://d22po4pjz3o32e.cloudfront.net/placeholder-image.svg"
alt="Relume placeholder image 1"
class="absolute inset-0 size-full object-cover"
/>
</div>
<div
role="group"
aria-roledescription="slide"
class="min-w-0 shrink-0 grow-0 basis-full relative h-dvh pl-0"
>
<img
src="https://d22po4pjz3o32e.cloudfront.net/placeholder-image.svg"
alt="Relume placeholder image 2"
class="absolute inset-0 size-full object-cover"
/>
</div>
</div>
</div>
<button
class="focus-visible:ring-border-primary gap-3 items-center justify-center whitespace-nowrap ring-offset-white transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 border border-border-primary text-text-primary absolute size-14 rounded-full bg-neutral-white left-0 top-1/2 -translate-y-1/2 -mt-8 ml-8 hidden lg:flex"
disabled=""
>
<svg
stroke="currentColor"
fill="currentColor"
stroke-width="0"
viewBox="0 0 24 24"
class="size-6"
height="1em"
width="1em"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M12.707 17.293 8.414 13H18v-2H8.414l4.293-4.293-1.414-1.414L4.586 12l6.707 6.707z"
></path></svg
><span class="sr-only">Previous slide</span></button
><button
class="focus-visible:ring-border-primary gap-3 items-center justify-center whitespace-nowrap ring-offset-white transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 border border-border-primary text-text-primary absolute size-14 rounded-full bg-neutral-white right-0 top-1/2 -translate-y-1/2 -mt-8 mr-8 hidden lg:flex"
disabled=""
>
<svg
stroke="currentColor"
fill="currentColor"
stroke-width="0"
viewBox="0 0 24 24"
class="size-6"
height="1em"
width="1em"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="m11.293 17.293 1.414 1.414L19.414 12l-6.707-6.707-1.414 1.414L15.586 11H6v2h9.586z"
></path></svg
><span class="sr-only">Next slide</span>
</button>
<div class="absolute bottom-8 left-1/2 -translate-x-1/2 transform">
<button class="relative mx-[3px] inline-block size-2 rounded-full bg-white/40"></button
><button class="relative mx-[3px] inline-block size-2 rounded-full bg-white/40"></button>
</div>
</div>
</section>
"use client";
import { useState, useEffect } from "react";
import {
Carousel,
CarouselContent,
CarouselItem,
CarouselNext,
CarouselPrevious,
} from "@relume_io/relume-ui";
import type { CarouselApi } from "@relume_io/relume-ui";
import clsx from "clsx";
type ImageProps = {
src: string;
alt?: string;
};
type Props = {
heading: string;
description: string;
images: ImageProps[];
};
export type Gallery13Props = React.ComponentPropsWithoutRef<"section"> & Partial<Props>;
export const Gallery13 = (props: Gallery13Props) => {
const { heading, description, images } = {
...Gallery13Defaults,
...props,
};
const [api, setApi] = useState<CarouselApi>();
const [current, setCurrent] = useState(0);
useEffect(() => {
if (!api) {
return;
}
setCurrent(api.selectedScrollSnap() + 1);
api.on("select", () => {
setCurrent(api.selectedScrollSnap() + 1);
});
}, [api]);
return (
<section id="relume">
<div className="px-[5%] py-16 md:py-24 lg:py-28">
<div className="container text-center">
<h2 className="rb-5 mb-5 text-5xl font-bold md:mb-6 md:text-7xl lg:text-8xl">
{heading}
</h2>
<p className="md:text-md">{description}</p>
</div>
</div>
{/* for all available options: https://www.embla-carousel.com/api/options/ */}
<Carousel
setApi={setApi}
opts={{
loop: true,
align: "start",
}}
className="overflow-hidden"
>
<CarouselContent className="ml-0">
{images.map((image, index) => (
<CarouselItem key={index} className="relative h-dvh pl-0">
<img
src={image.src}
alt={image.alt}
className="absolute inset-0 size-full object-cover"
/>
</CarouselItem>
))}
</CarouselContent>
<CarouselPrevious className="-mt-8 ml-8 hidden lg:flex" />
<CarouselNext className="-mt-8 mr-8 hidden lg:flex" />
<div className="absolute bottom-8 left-1/2 -translate-x-1/2 transform">
{images.map((_, index) => (
<button
key={index}
onClick={() => api?.scrollTo(index)}
className={clsx(
"relative mx-[3px] inline-block size-2 rounded-full",
current === index + 1 ? "bg-white" : "bg-white/40",
)}
/>
))}
</div>
</Carousel>
</section>
);
};
export const Gallery13Defaults: Props = {
heading: "Image Gallery",
description: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.",
images: [
{
src: "https://d22po4pjz3o32e.cloudfront.net/placeholder-image.svg",
alt: "Relume placeholder image 1",
},
{
src: "https://d22po4pjz3o32e.cloudfront.net/placeholder-image.svg",
alt: "Relume placeholder image 2",
},
],
};
Need help?
For installation guidelines and API information, visit the docs.