Files
gridpilot.gg/apps/website/components/landing/EmailCapture.tsx
2025-12-02 00:19:49 +01:00

172 lines
7.8 KiB
TypeScript

'use client';
import { useState, FormEvent } from 'react';
import { motion, AnimatePresence } from 'framer-motion';
export default function EmailCapture() {
const [email, setEmail] = useState('');
const [isValid, setIsValid] = useState(true);
const [submitted, setSubmitted] = useState(false);
const [isSubmitting, setIsSubmitting] = useState(false);
const [errorMessage, setErrorMessage] = useState('');
const handleSubmit = async (e: FormEvent<HTMLFormElement>) => {
e.preventDefault();
if (!email) {
setIsValid(false);
setErrorMessage('Email is required');
return;
}
setIsSubmitting(true);
setErrorMessage('');
try {
const response = await fetch('/api/signup', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ email }),
});
const data = await response.json();
if (!response.ok) {
setIsValid(false);
setErrorMessage(data.error || 'Failed to submit email');
return;
}
setSubmitted(true);
setEmail('');
} catch (error) {
setIsValid(false);
setErrorMessage('Network error. Please try again.');
console.error('Signup error:', error);
} finally {
setIsSubmitting(false);
}
};
return (
<section id="early-access" className="relative py-24 bg-gradient-to-b from-deep-graphite to-iron-gray">
<div className="max-w-2xl mx-auto px-6">
<AnimatePresence mode="wait">
{submitted ? (
<motion.div
key="success"
initial={{ opacity: 0, scale: 0.95 }}
animate={{ opacity: 1, scale: 1 }}
transition={{ duration: 0.3 }}
className="relative rounded-xl bg-gradient-to-br from-iron-gray via-deep-graphite to-iron-gray p-12 text-center border border-charcoal-outline shadow-[0_0_80px_rgba(111,227,122,0.15)]"
>
<motion.div
initial={{ scale: 0 }}
animate={{ scale: 1 }}
transition={{ delay: 0.2, type: 'spring', stiffness: 200 }}
className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-performance-green/20 mb-6"
>
<svg className="w-8 h-8 text-performance-green" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
</svg>
</motion.div>
<h2 className="text-3xl font-semibold text-white mb-4">You&apos;re on the list!</h2>
<p className="text-base text-gray-400 font-light">
Check your email for confirmation. We&apos;ll notify you when GridPilot launches.
</p>
</motion.div>
) : (
<motion.div
key="form"
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
className="relative rounded-xl bg-gradient-to-br from-iron-gray via-deep-graphite to-iron-gray p-8 md:p-16 border border-charcoal-outline shadow-[0_0_80px_rgba(25,140,255,0.15)]"
>
<div className="text-center mb-10">
<h2 className="text-4xl md:text-5xl font-semibold text-white mb-3">
Join the Waitlist
</h2>
<div className="w-32 h-1 bg-gradient-to-r from-primary-blue to-neon-aqua mx-auto mb-6 rounded-full" />
<p className="text-lg text-gray-400 font-light max-w-lg mx-auto">
Be among the first to experience GridPilot when we launch early access.
</p>
</div>
<form onSubmit={handleSubmit} className="space-y-6">
<div className="flex flex-col sm:flex-row gap-4">
<div className="flex-1">
<input
type="email"
value={email}
onChange={(e) => {
setEmail(e.target.value);
setIsValid(true);
setErrorMessage('');
}}
placeholder="your@email.com"
disabled={isSubmitting}
className={`w-full px-6 py-4 rounded-lg bg-iron-gray text-white placeholder-gray-500 border transition-all duration-150 ${
!isValid
? 'border-red-500 focus:ring-2 focus:ring-red-500'
: 'border-charcoal-outline focus:border-neon-aqua focus:ring-2 focus:ring-neon-aqua/50'
} hover:scale-[1.01] disabled:opacity-50 disabled:cursor-not-allowed`}
aria-label="Email address"
/>
{!isValid && errorMessage && (
<p className="mt-2 text-sm text-red-400">{errorMessage}</p>
)}
</div>
<button
type="submit"
disabled={isSubmitting}
className="px-8 py-4 rounded-full bg-gradient-to-r from-primary-blue to-neon-aqua text-white font-semibold transition-all duration-150 hover:shadow-glow-strong hover:scale-105 active:scale-95 disabled:opacity-50 disabled:cursor-not-allowed disabled:hover:scale-100 whitespace-nowrap"
>
{isSubmitting ? (
<span className="flex items-center gap-2">
<svg className="animate-spin h-5 w-5" viewBox="0 0 24 24">
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" fill="none" />
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z" />
</svg>
Joining...
</span>
) : (
'Get Early Access'
)}
</button>
</div>
</form>
<motion.div
initial={{ opacity: 0, y: 10 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.2 }}
className="mt-8 flex flex-col sm:flex-row gap-4 justify-center text-sm text-gray-400 font-light"
>
<div className="flex items-center gap-2">
<svg className="w-4 h-4 text-performance-green" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clipRule="evenodd" />
</svg>
<span>No spam, ever</span>
</div>
<div className="flex items-center gap-2">
<svg className="w-4 h-4 text-performance-green" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clipRule="evenodd" />
</svg>
<span>Early feature access</span>
</div>
<div className="flex items-center gap-2">
<svg className="w-4 h-4 text-performance-green" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clipRule="evenodd" />
</svg>
<span>Priority support</span>
</div>
</motion.div>
</motion.div>
)}
</AnimatePresence>
</div>
</section>
);
}