Initial commit

This commit is contained in:
dk
2026-07-01 20:26:56 +00:00
commit b01a6ae2fe
349 changed files with 43638 additions and 0 deletions

View File

@@ -0,0 +1,65 @@
import type { LucideIcon } from "lucide-react";
import TextAnimation from "@/components/ui/TextAnimation";
import ScrollReveal from "@/components/ui/ScrollReveal";
import { useButtonClick } from "@/hooks/useButtonClick";
type ContactOption = {
icon: LucideIcon;
label: string;
href: string;
};
type ContactBarProps = {
tag: string;
title: string;
options: ContactOption[];
textAnimation: "slide-up" | "fade-blur" | "fade";
};
const ContactOptionButton = ({ icon: Icon, label, href }: ContactOption) => {
const handleClick = useButtonClick(href);
return (
<button
onClick={handleClick}
className="flex flex-col items-center gap-3 group cursor-pointer"
>
<div className="size-22 md:size-16 lg:size-24 xl:size-28 2xl:size-30 rounded-full primary-button flex items-center justify-center">
<Icon className="size-4/10 text-primary-cta-text" strokeWidth={1.5} />
</div>
<p className="text-sm md:text-base text-foreground/60 group-hover:text-foreground transition-colors">{label}</p>
</button>
);
};
const ContactBar = ({ tag, title, options, textAnimation }: ContactBarProps) => {
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="fade-blur">
<div className="flex flex-col md:flex-row items-center justify-between gap-8 md:gap-10 card rounded p-6 md:p-10">
<div className="flex flex-col gap-2 md:max-w-1/2">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={title}
variant={textAnimation}
gradientText={true}
tag="h2"
className="text-7xl 2xl:text-8xl leading-[1.15] font-semibold text-balance"
/>
</div>
<div className="flex items-center gap-6 md:gap-10">
{options.map((opt) => (
<ContactOptionButton key={opt.label} {...opt} />
))}
</div>
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactBar;

View File

@@ -0,0 +1,101 @@
import { useState } from "react";
import ScrollReveal from "@/components/ui/ScrollReveal";
import TextAnimation from "@/components/ui/TextAnimation";
import { sendContactEmail } from "@/lib/api/email";
type ContactCenterProps = {
tag: string;
title: string;
description: string;
inputPlaceholder: string;
buttonText: string;
onSubmit?: (email: string) => void;
textAnimation: "slide-up" | "fade-blur" | "fade";
};
const ContactCenter = ({
tag,
title,
description,
inputPlaceholder,
buttonText,
onSubmit,
textAnimation,
}: ContactCenterProps) => {
const [email, setEmail] = useState("");
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const handleSubmit = async (e: React.SyntheticEvent<HTMLFormElement>) => {
e.preventDefault();
setIsLoading(true);
setError(null);
try {
await sendContactEmail({ email });
onSubmit?.(email);
setEmail("");
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to send. Please try again.");
} finally {
setIsLoading(false);
}
};
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="slide-up" className="flex items-center justify-center py-20 card rounded">
<div className="flex flex-col items-center w-full md:w-1/2 gap-2 px-5">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={title}
variant={textAnimation}
gradientText={true}
tag="h2"
className="md:max-w-9/10 text-6xl 2xl:text-7xl leading-[1.15] font-semibold text-center text-balance"
/>
<TextAnimation
text={description}
variant={textAnimation}
gradientText={false}
tag="p"
className="md:max-w-9/10 text-lg md:text-xl leading-snug text-center text-balance"
/>
<form
onSubmit={handleSubmit}
className="flex flex-col md:flex-row w-full md:w-8/10 2xl:w-6/10 gap-3 md:gap-1 p-1 mt-2 md:mt-3 card rounded"
>
<input
type="email"
placeholder={inputPlaceholder}
value={email}
onChange={(e) => setEmail(e.target.value)}
required
className="flex-1 px-5 py-3 md:py-0 text-base text-center md:text-left bg-transparent placeholder:opacity-75 focus:outline-none truncate"
aria-label="Email address"
/>
<button
type="submit"
disabled={isLoading}
className="flex items-center justify-center h-10 px-6 text-sm rounded primary-button text-primary-cta-text cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed"
>
{isLoading ? "Sending..." : buttonText}
</button>
</form>
{error && (
<p className="text-sm text-red-500 text-center">{error}</p>
)}
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactCenter;

View File

@@ -0,0 +1,48 @@
import ScrollReveal from "@/components/ui/ScrollReveal";
import TextAnimation from "@/components/ui/TextAnimation";
import Button from "@/components/ui/Button";
const ContactCta = ({
tag,
text,
primaryButton,
secondaryButton,
textAnimation,
}: {
tag: string;
text: string;
primaryButton: { text: string; href: string };
secondaryButton: { text: string; href: string };
textAnimation: "slide-up" | "fade-blur" | "fade";
}) => {
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="slide-up">
<div className="flex flex-col items-center gap-8 md:gap-10 py-20 px-8 rounded card">
<div className="flex flex-col items-center gap-2">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={text}
variant={textAnimation}
gradientText={true}
tag="h2"
className="md:max-w-8/10 text-5xl 2xl:text-6xl leading-[1.15] font-semibold text-center text-balance"
/>
<div className="flex flex-wrap justify-center gap-3 mt-2 md:mt-3">
<Button text={primaryButton.text} href={primaryButton.href} variant="primary" />
<Button text={secondaryButton.text} href={secondaryButton.href} variant="secondary" animationDelay={0.1} />
</div>
</div>
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactCta;

View File

@@ -0,0 +1,159 @@
import { useRef, useState } from "react";
import { useScroll, useTransform, motion } from "motion/react";
import ImageOrVideo from "@/components/ui/ImageOrVideo";
import { sendContactEmail } from "@/lib/api/email";
type ContactParallaxCardProps = {
title: string;
inputs: { name: string; type: string; placeholder: string; required?: boolean }[];
textarea?: { name: string; placeholder: string; rows?: number; required?: boolean };
buttonText: string;
footerText: string;
footerLink: { text: string; href: string; imageSrc: string };
onSubmit?: (data: Record<string, string>) => void;
} & ({ imageSrc: string; videoSrc?: never } | { videoSrc: string; imageSrc?: never });
const ContactParallaxCard = ({
title,
imageSrc,
videoSrc,
inputs,
textarea,
buttonText,
footerText,
footerLink,
onSubmit,
}: ContactParallaxCardProps) => {
const sectionRef = useRef<HTMLDivElement>(null);
const [formData, setFormData] = useState<Record<string, string>>(() => {
const initial: Record<string, string> = {};
inputs.forEach((input) => {
initial[input.name] = "";
});
if (textarea) {
initial[textarea.name] = "";
}
return initial;
});
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const handleSubmit = async (e: React.SyntheticEvent<HTMLFormElement>) => {
e.preventDefault();
setIsLoading(true);
setError(null);
try {
await sendContactEmail({ formData });
onSubmit?.(formData);
const reset: Record<string, string> = {};
inputs.forEach((input) => {
reset[input.name] = "";
});
if (textarea) {
reset[textarea.name] = "";
}
setFormData(reset);
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to send. Please try again.");
} finally {
setIsLoading(false);
}
};
const { scrollYProgress } = useScroll({
target: sectionRef,
offset: ["start end", "end start"],
});
const bgY = useTransform(scrollYProgress, [0, 1], ["-10%", "10%"]);
return (
<section
ref={sectionRef}
aria-label="Contact section"
className="relative overflow-hidden h-[90svh] my-20"
>
<motion.div className="absolute inset-0" style={{ y: bgY }}>
<ImageOrVideo
imageSrc={imageSrc}
videoSrc={videoSrc}
className="absolute inset-0 w-full h-[120%] object-cover rounded-none"
/>
<div className="absolute inset-0 bg-black/30" />
</motion.div>
<div className="relative z-10 flex items-center h-full px-8 md:px-12">
<div className="mx-auto w-content-width">
<motion.div
initial={{ opacity: 0, y: 30 }}
whileInView={{ opacity: 1, y: 0 }}
viewport={{ once: true }}
transition={{ duration: 0.7, ease: [0.22, 1, 0.36, 1] }}
className="w-full md:w-1/2 lg:w-5/12 rounded border border-primary-cta-text/10 bg-primary-cta-text/10 backdrop-blur-xl p-6 md:p-10"
>
<h2 className="mb-6 text-4xl md:text-5xl font-semibold text-white">
{title}
</h2>
<form onSubmit={handleSubmit} className="flex flex-col gap-4">
<div className="grid grid-cols-1 gap-4 md:grid-cols-2">
{inputs.map((input) => (
<input
key={input.name}
type={input.type}
placeholder={input.placeholder}
value={formData[input.name] || ""}
onChange={(e) => setFormData({ ...formData, [input.name]: e.target.value })}
required={input.required}
aria-label={input.placeholder}
className="w-full h-13 rounded border border-primary-cta-text/15 bg-primary-cta-text/10 px-5 text-base text-white placeholder:text-white/40 focus:border-accent/50 focus:outline-none"
/>
))}
</div>
{textarea && (
<textarea
placeholder={textarea.placeholder}
value={formData[textarea.name] || ""}
onChange={(e) => setFormData({ ...formData, [textarea.name]: e.target.value })}
required={textarea.required}
rows={textarea.rows || 5}
aria-label={textarea.placeholder}
className="w-full resize-none rounded border border-primary-cta-text/15 bg-primary-cta-text/10 px-5 py-3 text-base text-white placeholder:text-white/40 focus:border-accent/50 focus:outline-none"
/>
)}
<button
type="submit"
disabled={isLoading}
className="flex items-center justify-center w-full h-13 px-6 text-base font-medium rounded bg-background text-foreground cursor-pointer transition-colors hover:bg-background/90 disabled:opacity-50 disabled:cursor-not-allowed"
>
{isLoading ? "Sending..." : buttonText}
</button>
{error && (
<p className="text-sm text-red-500 text-center">{error}</p>
)}
</form>
<div className="mt-6 flex items-center justify-between border-t border-primary-cta-text/10 pt-6">
<p className="text-base text-white/75">{footerText}</p>
<a
href={footerLink.href}
className="flex items-center gap-2 rounded-full bg-primary-cta-text/15 backdrop-blur-sm p-1.5 pr-5 text-sm font-medium text-white whitespace-nowrap transition-colors hover:bg-primary-cta-text/25"
>
<ImageOrVideo
imageSrc={footerLink.imageSrc}
className="h-8 w-8 rounded-full object-cover"
/>
{footerLink.text}
</a>
</div>
</motion.div>
</div>
</div>
</section>
);
};
export default ContactParallaxCard;

View File

@@ -0,0 +1,110 @@
import { useState } from "react";
import ScrollReveal from "@/components/ui/ScrollReveal";
import TextAnimation from "@/components/ui/TextAnimation";
import ImageOrVideo from "@/components/ui/ImageOrVideo";
import { sendContactEmail } from "@/lib/api/email";
type ContactSplitEmailProps = {
tag: string;
title: string;
description: string;
inputPlaceholder: string;
buttonText: string;
onSubmit?: (email: string) => void;
textAnimation: "slide-up" | "fade-blur" | "fade";
} & ({ imageSrc: string; videoSrc?: never } | { videoSrc: string; imageSrc?: never });
const ContactSplitEmail = ({
tag,
title,
description,
inputPlaceholder,
buttonText,
onSubmit,
imageSrc,
videoSrc,
textAnimation,
}: ContactSplitEmailProps) => {
const [email, setEmail] = useState("");
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const handleSubmit = async (e: React.SyntheticEvent<HTMLFormElement>) => {
e.preventDefault();
setIsLoading(true);
setError(null);
try {
await sendContactEmail({ email });
onSubmit?.(email);
setEmail("");
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to send. Please try again.");
} finally {
setIsLoading(false);
}
};
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="slide-up" className="grid grid-cols-1 md:grid-cols-2 gap-5">
<div className="flex items-center justify-center py-15 md:py-20 card rounded">
<div className="flex flex-col items-center w-full gap-2 px-5">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={title}
variant={textAnimation}
gradientText={true}
tag="h2"
className="text-6xl 2xl:text-7xl leading-[1.15] font-semibold text-center text-balance"
/>
<TextAnimation
text={description}
variant={textAnimation}
gradientText={false}
tag="p"
className="md:max-w-8/10 text-lg md:text-xl leading-snug text-center text-balance"
/>
<form
onSubmit={handleSubmit}
className="flex flex-col md:flex-row w-full md:w-8/10 2xl:w-6/10 gap-3 md:gap-1 p-1 mt-2 md:mt-3 card rounded"
>
<input
type="email"
placeholder={inputPlaceholder}
value={email}
onChange={(e) => setEmail(e.target.value)}
required
className="flex-1 px-5 py-3 md:py-0 text-base text-center md:text-left bg-transparent placeholder:opacity-75 focus:outline-none truncate"
aria-label="Email address"
/>
<button
type="submit"
disabled={isLoading}
className="flex items-center justify-center h-10 px-6 text-sm rounded primary-button text-primary-cta-text cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed"
>
{isLoading ? "Sending..." : buttonText}
</button>
</form>
{error && (
<p className="text-sm text-red-500 text-center">{error}</p>
)}
</div>
</div>
<div className="h-100 md:h-auto md:aspect-square card rounded overflow-hidden">
<ImageOrVideo imageSrc={imageSrc} videoSrc={videoSrc} />
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactSplitEmail;

View File

@@ -0,0 +1,157 @@
import { useState } from "react";
import ScrollReveal from "@/components/ui/ScrollReveal";
import TextAnimation from "@/components/ui/TextAnimation";
import ImageOrVideo from "@/components/ui/ImageOrVideo";
import { sendContactEmail } from "@/lib/api/email";
type InputField = {
name: string;
type: string;
placeholder: string;
required?: boolean;
};
type TextareaField = {
name: string;
placeholder: string;
rows?: number;
required?: boolean;
};
type ContactSplitFormProps = {
tag: string;
title: string;
description: string;
inputs: InputField[];
textarea?: TextareaField;
buttonText: string;
onSubmit?: (data: Record<string, string>) => void;
textAnimation: "slide-up" | "fade-blur" | "fade";
} & ({ imageSrc: string; videoSrc?: never } | { videoSrc: string; imageSrc?: never });
const ContactSplitForm = ({
tag,
title,
description,
inputs,
textarea,
buttonText,
onSubmit,
imageSrc,
videoSrc,
textAnimation,
}: ContactSplitFormProps) => {
const [formData, setFormData] = useState<Record<string, string>>(() => {
const initial: Record<string, string> = {};
inputs.forEach((input) => {
initial[input.name] = "";
});
if (textarea) {
initial[textarea.name] = "";
}
return initial;
});
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const handleSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault();
setIsLoading(true);
setError(null);
try {
await sendContactEmail({ formData });
onSubmit?.(formData);
const reset: Record<string, string> = {};
inputs.forEach((input) => {
reset[input.name] = "";
});
if (textarea) {
reset[textarea.name] = "";
}
setFormData(reset);
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to send. Please try again.");
} finally {
setIsLoading(false);
}
};
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="fade" className="grid grid-cols-1 md:grid-cols-2 gap-5">
<div className="p-6 md:p-10 card rounded">
<form onSubmit={handleSubmit} className="flex flex-col gap-6">
<div className="flex flex-col items-center gap-2 text-center">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={title}
variant={textAnimation}
gradientText={true}
tag="h2"
className="text-6xl 2xl:text-7xl leading-[1.15] font-semibold text-balance"
/>
<TextAnimation
text={description}
variant={textAnimation}
gradientText={false}
tag="p"
className="text-lg md:text-xl leading-snug text-balance"
/>
</div>
<div className="flex flex-col gap-3">
{inputs.map((input) => (
<input
key={input.name}
type={input.type}
placeholder={input.placeholder}
value={formData[input.name] || ""}
onChange={(e) => setFormData({ ...formData, [input.name]: e.target.value })}
required={input.required}
aria-label={input.placeholder}
className="w-full px-5 py-3 text-base bg-transparent placeholder:opacity-75 focus:outline-none card rounded"
/>
))}
{textarea && (
<textarea
placeholder={textarea.placeholder}
value={formData[textarea.name] || ""}
onChange={(e) => setFormData({ ...formData, [textarea.name]: e.target.value })}
required={textarea.required}
rows={textarea.rows || 5}
aria-label={textarea.placeholder}
className="w-full px-5 py-3 text-base bg-transparent placeholder:opacity-75 focus:outline-none resize-none card rounded"
/>
)}
<button
type="submit"
disabled={isLoading}
className="flex items-center justify-center w-full h-10 px-6 text-sm primary-button text-primary-cta-text rounded cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed"
>
{isLoading ? "Sending..." : buttonText}
</button>
{error && (
<p className="text-sm text-red-500 text-center">{error}</p>
)}
</div>
</form>
</div>
<div className="h-100 md:h-auto card rounded overflow-hidden">
<ImageOrVideo imageSrc={imageSrc} videoSrc={videoSrc} className="size-full object-cover rounded" />
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactSplitForm;

View File

@@ -0,0 +1,204 @@
import { useRef, useState } from "react";
import { useScroll, useTransform, motion } from "motion/react";
import type { LucideIcon } from "lucide-react";
import ImageOrVideo from "@/components/ui/ImageOrVideo";
import TextAnimation from "@/components/ui/TextAnimation";
import ScrollReveal from "@/components/ui/ScrollReveal";
import { sendContactEmail } from "@/lib/api/email";
import { useButtonClick } from "@/hooks/useButtonClick";
import { resolveIcon } from "@/utils/resolve-icon";
type InputField = {
name: string;
type: string;
placeholder: string;
required?: boolean;
};
type TextareaField = {
name: string;
placeholder: string;
rows?: number;
required?: boolean;
};
type CtaLink = {
icon: string | LucideIcon;
label: string;
href?: string;
onClick?: () => void;
};
type ContactSplitFormParallaxProps = {
tag: string;
title: string;
description: string;
inputs: InputField[];
textarea?: TextareaField;
buttonText: string;
onSubmit?: (data: Record<string, string>) => void;
ctaLinks?: CtaLink[];
textAnimation: "slide-up" | "fade-blur" | "fade";
} & ({ imageSrc: string; videoSrc?: never } | { videoSrc: string; imageSrc?: never });
const CtaLinkButton = ({ icon, label, href, onClick }: CtaLink) => {
const Icon = resolveIcon(icon);
const handleClick = useButtonClick(href, onClick);
return (
<a
href={href}
onClick={handleClick}
className="flex items-center justify-center gap-2 h-9 px-3 text-sm rounded-full cursor-pointer backdrop-blur-xl bg-primary-cta-text/15 border border-primary-cta-text/20 text-primary-cta-text font-semibold hover:bg-primary-cta-text/25 transition-all duration-300 ease-out"
>
<Icon className="size-4" strokeWidth={1.5} />
<span>{label}</span>
</a>
);
};
const ContactSplitFormParallax = ({
tag,
title,
description,
inputs,
textarea,
buttonText,
onSubmit,
imageSrc,
videoSrc,
ctaLinks,
textAnimation,
}: ContactSplitFormParallaxProps) => {
const imageRef = useRef<HTMLDivElement>(null);
const [formData, setFormData] = useState<Record<string, string>>(() => {
const initial: Record<string, string> = {};
inputs.forEach((input) => {
initial[input.name] = "";
});
if (textarea) {
initial[textarea.name] = "";
}
return initial;
});
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const handleSubmit = async (e: React.SyntheticEvent<HTMLFormElement>) => {
e.preventDefault();
setIsLoading(true);
setError(null);
try {
await sendContactEmail({ formData });
onSubmit?.(formData);
const reset: Record<string, string> = {};
inputs.forEach((input) => {
reset[input.name] = "";
});
if (textarea) {
reset[textarea.name] = "";
}
setFormData(reset);
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to send. Please try again.");
} finally {
setIsLoading(false);
}
};
const { scrollYProgress } = useScroll({
target: imageRef,
offset: ["start end", "end start"],
});
const imageScale = useTransform(scrollYProgress, [0, 0.6], [1.3, 1]);
return (
<section aria-label="Contact section" className="py-20">
<div className="w-content-width mx-auto">
<ScrollReveal variant="fade-blur" className="grid grid-cols-1 md:grid-cols-2 gap-5">
<div className="p-6 md:p-10 card rounded">
<form onSubmit={handleSubmit} className="flex flex-col gap-6">
<div className="flex flex-col items-center gap-2 text-center">
<div className="px-3 py-1 mb-1 text-sm card rounded w-fit">
<p>{tag}</p>
</div>
<TextAnimation
text={title}
variant={textAnimation}
gradientText={true}
tag="h2"
className="text-6xl 2xl:text-7xl leading-[1.15] font-semibold text-balance"
/>
<TextAnimation
text={description}
variant={textAnimation}
gradientText={false}
tag="p"
className="text-lg md:text-xl leading-snug text-balance"
/>
</div>
<div className="flex flex-col gap-3">
{inputs.map((input) => (
<input
key={input.name}
type={input.type}
placeholder={input.placeholder}
value={formData[input.name] || ""}
onChange={(e) => setFormData({ ...formData, [input.name]: e.target.value })}
required={input.required}
aria-label={input.placeholder}
className="w-full px-5 py-3 text-base bg-transparent placeholder:opacity-75 focus:outline-none card rounded"
/>
))}
{textarea && (
<textarea
placeholder={textarea.placeholder}
value={formData[textarea.name] || ""}
onChange={(e) => setFormData({ ...formData, [textarea.name]: e.target.value })}
required={textarea.required}
rows={textarea.rows || 5}
aria-label={textarea.placeholder}
className="w-full px-5 py-3 text-base bg-transparent placeholder:opacity-75 focus:outline-none resize-none card rounded"
/>
)}
<button
type="submit"
disabled={isLoading}
className="flex items-center justify-center w-full h-10 px-6 text-sm rounded primary-button text-primary-cta-text cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed"
>
{isLoading ? "Sending..." : buttonText}
</button>
{error && (
<p className="text-sm text-red-500 text-center">{error}</p>
)}
</div>
</form>
</div>
<div ref={imageRef} className="relative h-100 md:h-auto card rounded overflow-hidden">
<motion.div style={{ scale: imageScale }} className="w-full h-full origin-center">
<ImageOrVideo imageSrc={imageSrc} videoSrc={videoSrc} className="md:absolute md:inset-0 size-full object-cover" />
</motion.div>
{ctaLinks && ctaLinks.length > 0 && (
<div className="absolute inset-0 flex flex-wrap items-end justify-center gap-3 p-6 xl:p-7 2xl:p-8">
{ctaLinks.map((link, index) => (
<CtaLinkButton key={index} {...link} />
))}
</div>
)}
</div>
</ScrollReveal>
</div>
</section>
);
};
export default ContactSplitFormParallax;