182 lines
8.3 KiB
TypeScript
182 lines
8.3 KiB
TypeScript
|
|
"use client";
|
||
|
|
|
||
|
|
import { motion, useInView } from "framer-motion";
|
||
|
|
import { useEffect, useRef, useState } from "react";
|
||
|
|
import { Send } from "lucide-react";
|
||
|
|
import { Button } from "@/components/ui/button";
|
||
|
|
import { Input } from "@/components/ui/input";
|
||
|
|
import { Textarea } from "@/components/ui/textarea";
|
||
|
|
import { Card, CardContent } from "@/components/ui/card";
|
||
|
|
import { toast } from "sonner";
|
||
|
|
|
||
|
|
export function ContactSection() {
|
||
|
|
const ref = useRef(null);
|
||
|
|
const isInView = useInView(ref, { once: true, margin: "-100px" });
|
||
|
|
const [isDark, setIsDark] = useState(false);
|
||
|
|
const [formData, setFormData] = useState({
|
||
|
|
name: "",
|
||
|
|
email: "",
|
||
|
|
phone: "",
|
||
|
|
message: "",
|
||
|
|
});
|
||
|
|
|
||
|
|
// Sync with global theme class like Navbar/Hero/About
|
||
|
|
useEffect(() => {
|
||
|
|
const sync = () => setIsDark(document.documentElement.classList.contains("dark"));
|
||
|
|
sync();
|
||
|
|
const observer = new MutationObserver(sync);
|
||
|
|
observer.observe(document.documentElement, { attributes: true, attributeFilter: ["class"] });
|
||
|
|
return () => observer.disconnect();
|
||
|
|
}, []);
|
||
|
|
|
||
|
|
const handleSubmit = (e: React.FormEvent) => {
|
||
|
|
e.preventDefault();
|
||
|
|
toast("Message Received", {
|
||
|
|
description: "Thank you for reaching out. We'll get back to you soon!",
|
||
|
|
});
|
||
|
|
setFormData({ name: "", email: "", phone: "", message: "" });
|
||
|
|
};
|
||
|
|
|
||
|
|
return (
|
||
|
|
<section id="contact" className="relative overflow-hidden py-20" ref={ref}>
|
||
|
|
{/* Theme sync like Hero/About/Navbar */}
|
||
|
|
<div className="absolute inset-0 z-0" style={{ backgroundColor: isDark ? "#1a1e26" : "#ffffff" }} />
|
||
|
|
{!isDark && (
|
||
|
|
<div className="absolute inset-0 z-[1] bg-gradient-to-br from-rose-50/40 via-pink-50/40 to-orange-50/40" />
|
||
|
|
)}
|
||
|
|
{isDark && (
|
||
|
|
<div className="absolute inset-0 z-[1] bg-gradient-to-br from-gray-900/70 via-gray-800/70 to-gray-900/70" />
|
||
|
|
)}
|
||
|
|
|
||
|
|
{/* Subtle animated blobs */}
|
||
|
|
<div className="absolute inset-0 z-[2] overflow-hidden">
|
||
|
|
<motion.div
|
||
|
|
className={`absolute -top-10 left-10 h-64 w-64 rounded-full blur-3xl ${isDark ? "bg-pink-900/30 opacity-60" : "bg-pink-100 opacity-50"}`}
|
||
|
|
animate={{ x: [0, 60, 0], y: [0, 40, 0], scale: [1, 1.05, 1] }}
|
||
|
|
transition={{ duration: 18, repeat: Infinity, ease: "easeInOut" }}
|
||
|
|
/>
|
||
|
|
<motion.div
|
||
|
|
className={`absolute -bottom-10 right-10 h-72 w-72 rounded-full blur-3xl ${isDark ? "bg-orange-900/30 opacity-60" : "bg-orange-100 opacity-50"}`}
|
||
|
|
animate={{ x: [0, -60, 0], y: [0, -40, 0], scale: [1, 1.08, 1] }}
|
||
|
|
transition={{ duration: 22, repeat: Infinity, ease: "easeInOut" }}
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<div className="container mx-auto px-4 relative z-10 text-foreground">
|
||
|
|
<motion.div
|
||
|
|
initial={{ opacity: 0, y: 30 }}
|
||
|
|
animate={isInView ? { opacity: 1, y: 0 } : {}}
|
||
|
|
transition={{ duration: 0.6 }}
|
||
|
|
className="mb-16 text-center"
|
||
|
|
>
|
||
|
|
<h2 className="text-4xl md:text-5xl font-bold mb-4 bg-gradient-to-r from-rose-600 via-pink-600 to-orange-600 bg-clip-text text-transparent">
|
||
|
|
Get in Touch
|
||
|
|
</h2>
|
||
|
|
<div className="mx-auto mb-6 h-1 w-24 rounded-full bg-gradient-to-r from-rose-500 to-pink-600" />
|
||
|
|
<p className="mx-auto max-w-2xl text-lg text-muted-foreground">
|
||
|
|
Ready to start your journey? Reach out to schedule a consultation.
|
||
|
|
</p>
|
||
|
|
</motion.div>
|
||
|
|
|
||
|
|
<div className="grid gap-12 lg:grid-cols-2">
|
||
|
|
{/* Left: Illustration replacing cards */}
|
||
|
|
<motion.div
|
||
|
|
initial={{ opacity: 0, x: -50 }}
|
||
|
|
animate={isInView ? { opacity: 1, x: 0 } : {}}
|
||
|
|
transition={{ duration: 0.6, delay: 0.2 }}
|
||
|
|
className="relative"
|
||
|
|
>
|
||
|
|
<Card className="bg-card/60 backdrop-blur-sm border border-border/50 overflow-hidden">
|
||
|
|
<CardContent className="p-0">
|
||
|
|
{/* Use a high-quality, license-friendly illustration */}
|
||
|
|
<div className="relative">
|
||
|
|
<img
|
||
|
|
src="/3786819.jpg"
|
||
|
|
alt="Contact illustration"
|
||
|
|
className="mx-auto w-full max-w-xl p-6 select-none rounded-xl object-cover"
|
||
|
|
loading="lazy"
|
||
|
|
/>
|
||
|
|
<motion.div
|
||
|
|
className="pointer-events-none absolute inset-0"
|
||
|
|
animate={{ opacity: [0.3, 0.6, 0.3] }}
|
||
|
|
transition={{ duration: 3, repeat: Infinity }}
|
||
|
|
style={{ background: "radial-gradient(600px circle at 20% 10%, rgba(244,114,182,0.15), transparent 40%), radial-gradient(600px circle at 80% 80%, rgba(251,146,60,0.12), transparent 40%)" }}
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
<div className="px-6 pb-6">
|
||
|
|
<h3 className="text-2xl font-bold mb-2 text-foreground">Let's Begin Your Healing Journey</h3>
|
||
|
|
<p className="text-muted-foreground leading-relaxed">
|
||
|
|
Taking the first step toward therapy can feel daunting, but you're not alone. I'm here to support
|
||
|
|
you through every stage of your journey toward wellness and growth.
|
||
|
|
</p>
|
||
|
|
</div>
|
||
|
|
</CardContent>
|
||
|
|
</Card>
|
||
|
|
</motion.div>
|
||
|
|
|
||
|
|
{/* Right: Contact form */}
|
||
|
|
<motion.div
|
||
|
|
initial={{ opacity: 0, x: 50 }}
|
||
|
|
animate={isInView ? { opacity: 1, x: 0 } : {}}
|
||
|
|
transition={{ duration: 0.6, delay: 0.4 }}
|
||
|
|
>
|
||
|
|
<Card className="border border-border/50 bg-card/70 backdrop-blur-sm">
|
||
|
|
<CardContent className="p-6 md:p-8">
|
||
|
|
<h3 className="mb-6 text-2xl font-bold text-foreground">Send a Message</h3>
|
||
|
|
<form onSubmit={handleSubmit} className="space-y-4">
|
||
|
|
<div>
|
||
|
|
<Input
|
||
|
|
placeholder="Your Name"
|
||
|
|
value={formData.name}
|
||
|
|
onChange={(e) => setFormData({ ...formData, name: e.target.value })}
|
||
|
|
required
|
||
|
|
className="bg-card text-foreground dark:text-zinc-100 placeholder:text-muted-foreground dark:placeholder:text-zinc-400 border-border focus-visible:ring-rose-500"
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
<div>
|
||
|
|
<Input
|
||
|
|
type="email"
|
||
|
|
placeholder="Email Address"
|
||
|
|
value={formData.email}
|
||
|
|
onChange={(e) => setFormData({ ...formData, email: e.target.value })}
|
||
|
|
required
|
||
|
|
className="bg-card text-foreground dark:text-zinc-100 placeholder:text-muted-foreground dark:placeholder:text-zinc-400 border-border focus-visible:ring-rose-500"
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
<div>
|
||
|
|
<Input
|
||
|
|
type="tel"
|
||
|
|
placeholder="Phone Number"
|
||
|
|
value={formData.phone}
|
||
|
|
onChange={(e) => setFormData({ ...formData, phone: e.target.value })}
|
||
|
|
className="bg-card text-foreground dark:text-zinc-100 placeholder:text-muted-foreground dark:placeholder:text-zinc-400 border-border focus-visible:ring-rose-500"
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
<div>
|
||
|
|
<Textarea
|
||
|
|
placeholder="Tell me a bit about what brings you to therapy..."
|
||
|
|
rows={6}
|
||
|
|
value={formData.message}
|
||
|
|
onChange={(e) => setFormData({ ...formData, message: e.target.value })}
|
||
|
|
required
|
||
|
|
className="bg-card text-foreground dark:text-zinc-100 placeholder:text-muted-foreground dark:placeholder:text-zinc-400 border-border focus-visible:ring-rose-500"
|
||
|
|
/>
|
||
|
|
</div>
|
||
|
|
<Button
|
||
|
|
type="submit"
|
||
|
|
className="w-full cursor-pointer bg-gradient-to-r from-rose-500 to-pink-600 text-white transition-all hover:from-rose-600 hover:to-pink-700 hover:scale-[1.02]"
|
||
|
|
size="lg"
|
||
|
|
>
|
||
|
|
<Send className="mr-2 h-5 w-5" />
|
||
|
|
Send Message
|
||
|
|
</Button>
|
||
|
|
</form>
|
||
|
|
</CardContent>
|
||
|
|
</Card>
|
||
|
|
</motion.div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</section>
|
||
|
|
);
|
||
|
|
}
|