- Add animated hero section with parallax rocket SVG that descends on scroll - Add floating decorative particles and gradient layers in hero - Add staggered text reveal animation on hero h1 - Create ScrollReveal component (IntersectionObserver-based fade/slide) - Create AnimatedCounter component for stat numbers - Add scroll animations to all sections (Problematique, System, Demos, AboutMe, FAQ, Contact, Footer) - Add smooth FAQ accordion transitions - Add extensive CSS keyframe animations (float, flame, particles, stat glow) https://claude.ai/code/session_01V8YAjpqRQ3bfBYsABYsEgo
141 lines
6.9 KiB
TypeScript
141 lines
6.9 KiB
TypeScript
"use client";
|
|
|
|
import Image from "next/image";
|
|
import { urlFor } from "@/lib/sanity/client";
|
|
import type { SiteSettings } from "@/lib/sanity/queries";
|
|
import ScrollReveal from "@/components/animations/ScrollReveal";
|
|
import AnimatedCounter from "@/components/animations/AnimatedCounter";
|
|
|
|
interface AboutMeProps {
|
|
settings?: SiteSettings | null;
|
|
}
|
|
|
|
export default function AboutMe({ settings }: AboutMeProps) {
|
|
const name = settings?.ownerName || "Enguerrand";
|
|
const bio = settings?.ownerBio;
|
|
const address = settings?.address || "Flines-lez-Raches, Nord (59)";
|
|
const lat = settings?.lat || 50.4267;
|
|
const lng = settings?.lng || 3.2372;
|
|
const photoUrl = settings?.ownerPhoto ? urlFor(settings.ownerPhoto)?.width(400).height(480).url() : null;
|
|
|
|
return (
|
|
<section id="qui-suis-je" className="py-16 md:py-24 bg-bg" aria-label="Qui suis-je">
|
|
<div className="max-w-6xl mx-auto px-4 sm:px-6 lg:px-8">
|
|
{/* Header */}
|
|
<ScrollReveal direction="up">
|
|
<div className="text-center mb-12">
|
|
<span className="inline-block px-3 py-1.5 bg-navy/5 border border-navy/10 rounded-full text-navy text-xs font-semibold mb-4">
|
|
Votre expert local
|
|
</span>
|
|
<h2 className="text-2xl md:text-3xl lg:text-4xl font-bold text-navy tracking-[-0.02em]">
|
|
Pas une plateforme anonyme.{" "}
|
|
<span className="text-orange">Un voisin.</span>
|
|
</h2>
|
|
</div>
|
|
</ScrollReveal>
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-10 items-center">
|
|
{/* Left - Photo */}
|
|
<ScrollReveal direction="left">
|
|
<div className="flex justify-center">
|
|
<div className="relative">
|
|
<div className="w-64 h-72 sm:w-72 sm:h-80 bg-bg-muted border-2 border-border rounded-2xl flex items-center justify-center overflow-hidden">
|
|
{photoUrl ? (
|
|
<Image
|
|
src={photoUrl}
|
|
alt={`Photo de ${name}`}
|
|
fill
|
|
className="object-cover"
|
|
sizes="(max-width: 640px) 256px, 288px"
|
|
/>
|
|
) : (
|
|
<div className="text-center p-6">
|
|
<div className="w-20 h-20 bg-navy/10 rounded-full flex items-center justify-center mx-auto mb-4">
|
|
<svg className="w-10 h-10 text-navy/40" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z" />
|
|
</svg>
|
|
</div>
|
|
<p className="text-text-muted text-sm">Votre photo ici</p>
|
|
<p className="text-text-muted text-xs mt-1">(configurable via Sanity)</p>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className="absolute -bottom-3 left-1/2 -translate-x-1/2 bg-orange text-white text-xs font-bold px-4 py-1.5 rounded-full shadow-md whitespace-nowrap">
|
|
Basé à {address.split(",")[0]}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</ScrollReveal>
|
|
|
|
{/* Right - Text */}
|
|
<ScrollReveal direction="right">
|
|
<div>
|
|
{bio ? (
|
|
<p className="text-text text-base sm:text-lg leading-relaxed mb-4">
|
|
{bio}
|
|
</p>
|
|
) : (
|
|
<>
|
|
<p className="text-text text-base sm:text-lg leading-relaxed mb-4">
|
|
Je suis <strong className="text-navy">{name}</strong>, spécialisé dans la
|
|
visibilité locale et la construction de{" "}
|
|
<strong className="text-navy">systèmes de confiance en ligne</strong>{" "}
|
|
pour les TPE/PME du Nord.
|
|
</p>
|
|
<p className="text-text-light text-base leading-relaxed mb-4">
|
|
Je ne suis pas un call center parisien. Je connais la réalité de vos
|
|
chantiers à Douai, Orchies ou Valenciennes. Je sais que vous n’avez pas
|
|
le temps de gérer “un truc internet” et que vous voulez des résultats
|
|
concrets : des appels de <strong>vrais</strong> clients.
|
|
</p>
|
|
</>
|
|
)}
|
|
<p className="text-text-light text-base leading-relaxed mb-6">
|
|
Mon approche : je vous construis un <strong className="text-navy">dossier de confiance</strong>{" "}
|
|
(Google + site + preuves) qui transforme votre bouche-à-oreille en système
|
|
permanent. Pas de jargon, pas de blabla — du concret.
|
|
</p>
|
|
|
|
<div className="grid grid-cols-2 gap-4">
|
|
<div className="bg-bg-white border border-border rounded-xl p-4 text-center">
|
|
<p className="text-2xl font-bold text-navy">
|
|
<AnimatedCounter end={100} suffix="%" />
|
|
</p>
|
|
<p className="text-text-muted text-xs mt-1">Local Nord</p>
|
|
</div>
|
|
<div className="bg-bg-white border border-border rounded-xl p-4 text-center">
|
|
<p className="text-2xl font-bold text-navy">
|
|
<AnimatedCounter end={24} suffix="h" />
|
|
</p>
|
|
<p className="text-text-muted text-xs mt-1">Délai de réponse</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</ScrollReveal>
|
|
</div>
|
|
|
|
{/* Map */}
|
|
<ScrollReveal direction="up" className="mt-12">
|
|
<div className="bg-bg-white border border-border rounded-2xl overflow-hidden">
|
|
<div className="p-4 border-b border-border flex items-center gap-2">
|
|
<svg className="w-5 h-5 text-orange" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z" />
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 11a3 3 0 11-6 0 3 3 0 016 0z" />
|
|
</svg>
|
|
<span className="text-navy font-semibold text-sm">Zone d’intervention : Douai, Orchies, Arleux, Valenciennes et environs</span>
|
|
</div>
|
|
<div className="relative h-48 sm:h-64">
|
|
<iframe
|
|
src={`https://www.openstreetmap.org/export/embed.html?bbox=${lng - 0.45}%2C${lat - 0.18}%2C${lng + 0.45}%2C${lat + 0.12}&layer=mapnik&marker=${lat}%2C${lng}`}
|
|
className="absolute inset-0 w-full h-full border-0"
|
|
title={`Carte de localisation - ${address.split(",")[0]}`}
|
|
loading="lazy"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</ScrollReveal>
|
|
</div>
|
|
</section>
|
|
);
|
|
}
|