169 lines
6.2 KiB
TypeScript
169 lines
6.2 KiB
TypeScript
'use client';
|
|
|
|
import { useState } from 'react';
|
|
import { motion, AnimatePresence } from 'framer-motion';
|
|
import { X, ChevronLeft, ChevronRight, Maximize2 } from 'lucide-react';
|
|
import Image from 'next/image';
|
|
|
|
interface VillaGalleryProps {
|
|
images: string[];
|
|
name: string;
|
|
}
|
|
|
|
export default function VillaGallery({ images, name }: VillaGalleryProps) {
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
const [currentIndex, setCurrentIndex] = useState(0);
|
|
|
|
const openLightbox = (index: number) => {
|
|
setCurrentIndex(index);
|
|
setIsOpen(true);
|
|
document.body.style.overflow = 'hidden';
|
|
};
|
|
|
|
const closeLightbox = () => {
|
|
setIsOpen(false);
|
|
document.body.style.overflow = 'auto';
|
|
};
|
|
|
|
const nextImage = () => {
|
|
setCurrentIndex((prev) => (prev + 1) % images.length);
|
|
};
|
|
|
|
const prevImage = () => {
|
|
setCurrentIndex((prev) => (prev - 1 + images.length) % images.length);
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<div className="grid grid-cols-1 md:grid-cols-4 grid-rows-2 h-[400px] md:h-[600px] gap-4 rounded-3xl overflow-hidden shadow-2xl">
|
|
{/* Main large image */}
|
|
<div
|
|
className="md:col-span-2 md:row-span-2 relative group overflow-hidden cursor-zoom-in"
|
|
onClick={() => openLightbox(0)}
|
|
>
|
|
<Image
|
|
src={images[0]}
|
|
fill
|
|
className="object-cover transition-transform duration-700 group-hover:scale-105"
|
|
alt={`${name} 1`}
|
|
priority
|
|
/>
|
|
<div className="absolute inset-0 bg-black/20 opacity-0 group-hover:opacity-100 transition-opacity flex items-center justify-center">
|
|
<Maximize2 className="text-white w-8 h-8" />
|
|
</div>
|
|
</div>
|
|
|
|
{/* Second image */}
|
|
<div
|
|
className="md:col-span-1 relative group overflow-hidden cursor-zoom-in"
|
|
onClick={() => openLightbox(1)}
|
|
>
|
|
{images[1] ? (
|
|
<Image src={images[1]} fill className="object-cover transition-transform duration-700 group-hover:scale-105" alt={`${name} 2`} />
|
|
) : (
|
|
<div className="w-full h-full bg-aegean-dark/5" />
|
|
)}
|
|
<div className="absolute inset-0 bg-black/20 opacity-0 group-hover:opacity-100 transition-opacity flex items-center justify-center">
|
|
<Maximize2 className="text-white w-6 h-6" />
|
|
</div>
|
|
</div>
|
|
|
|
{/* Third image */}
|
|
<div
|
|
className="md:col-span-1 relative group overflow-hidden cursor-zoom-in"
|
|
onClick={() => openLightbox(2)}
|
|
>
|
|
{images[2] ? (
|
|
<Image src={images[2]} fill className="object-cover transition-transform duration-700 group-hover:scale-105" alt={`${name} 3`} />
|
|
) : (
|
|
<div className="w-full h-full bg-aegean-dark/5" />
|
|
)}
|
|
<div className="absolute inset-0 bg-black/20 opacity-0 group-hover:opacity-100 transition-opacity flex items-center justify-center">
|
|
<Maximize2 className="text-white w-6 h-6" />
|
|
</div>
|
|
</div>
|
|
|
|
{/* Fourth image with overlay if more exists */}
|
|
<div
|
|
className="md:col-span-2 relative group overflow-hidden cursor-zoom-in"
|
|
onClick={() => openLightbox(3)}
|
|
>
|
|
{images[3] ? (
|
|
<Image src={images[3]} fill className="object-cover transition-transform duration-700 group-hover:scale-105" alt={`${name} 4`} />
|
|
) : (
|
|
<div className="w-full h-full bg-aegean-dark/5" />
|
|
)}
|
|
|
|
{images.length > 4 && (
|
|
<div className="absolute inset-0 bg-black/50 flex flex-col items-center justify-center text-white backdrop-blur-[2px] group-hover:bg-black/40 transition-all">
|
|
<span className="text-3xl font-serif">+{images.length - 4}</span>
|
|
<span className="text-[10px] font-bold uppercase tracking-widest mt-2">Daha Fazla Fotoğraf</span>
|
|
</div>
|
|
)}
|
|
|
|
<div className="absolute inset-0 bg-black/20 opacity-0 group-hover:opacity-100 transition-opacity flex items-center justify-center">
|
|
{! (images.length > 4) && <Maximize2 className="text-white w-8 h-8" />}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Lightbox Overlay */}
|
|
<AnimatePresence>
|
|
{isOpen && (
|
|
<motion.div
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
exit={{ opacity: 0 }}
|
|
className="fixed inset-0 z-[100] bg-black/95 flex items-center justify-center p-4 md:p-8"
|
|
>
|
|
<button
|
|
onClick={closeLightbox}
|
|
className="absolute top-8 right-8 text-white/70 hover:text-white z-50 transition-colors"
|
|
>
|
|
<X className="w-8 h-8" />
|
|
</button>
|
|
|
|
<button
|
|
onClick={prevImage}
|
|
className="absolute left-4 md:left-8 text-white/70 hover:text-white z-50 p-2 rounded-full hover:bg-white/10 transition-all"
|
|
>
|
|
<ChevronLeft className="w-10 h-10" />
|
|
</button>
|
|
|
|
<button
|
|
onClick={nextImage}
|
|
className="absolute right-4 md:right-8 text-white/70 hover:text-white z-50 p-2 rounded-full hover:bg-white/10 transition-all"
|
|
>
|
|
<ChevronRight className="w-10 h-10" />
|
|
</button>
|
|
|
|
<motion.div
|
|
key={currentIndex}
|
|
initial={{ opacity: 0, scale: 0.9 }}
|
|
animate={{ opacity: 1, scale: 1 }}
|
|
exit={{ opacity: 0, scale: 0.9 }}
|
|
className="relative w-full h-full max-w-6xl max-h-[80vh]"
|
|
>
|
|
{images[currentIndex] ? (
|
|
<Image
|
|
src={images[currentIndex]}
|
|
fill
|
|
className="object-contain"
|
|
alt={`${name} Full`}
|
|
quality={100}
|
|
/>
|
|
) : (
|
|
<div className="flex items-center justify-center h-full text-white/50">Görsel Yüklenemedi</div>
|
|
)}
|
|
</motion.div>
|
|
|
|
<div className="absolute bottom-8 left-1/2 -translate-x-1/2 text-white/60 text-sm font-medium tracking-widest uppercase">
|
|
{currentIndex + 1} / {images.length} — {name}
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
</AnimatePresence>
|
|
</>
|
|
);
|
|
}
|