|
|
|
@@ -1,88 +1,365 @@ |
|
|
|
'use client';
|
|
|
|
import { useState, useEffect } from 'react';
|
|
|
|
import { Card } from '@/types';
|
|
|
|
|
|
|
|
export default function PublicGrid({ cards, maxCols = 5 }: { cards: Card[], maxCols?: number }) {
|
|
|
|
const [activeCard, setActiveCard] = useState<Card | null>(null);
|
|
|
|
|
|
|
|
// Prevent background scrolling when modal is open
|
|
|
|
useEffect(() => {
|
|
|
|
if (activeCard) document.body.style.overflow = 'hidden';
|
|
|
|
else document.body.style.overflow = 'unset';
|
|
|
|
return () => { document.body.style.overflow = 'unset'; }
|
|
|
|
}, [activeCard]);
|
|
|
|
|
|
|
|
// Tailwind classes mapping based on the admin's chosen max columns
|
|
|
|
const gridClasses: Record<number, string> = {
|
|
|
|
3: 'xl:grid-cols-3 lg:grid-cols-3 md:grid-cols-2 grid-cols-1', // ADDED THIS LINE
|
|
|
|
4: 'xl:grid-cols-4 lg:grid-cols-3 md:grid-cols-2 grid-cols-1',
|
|
|
|
5: 'xl:grid-cols-5 lg:grid-cols-4 md:grid-cols-3 grid-cols-1',
|
|
|
|
6: 'xl:grid-cols-6 lg:grid-cols-4 md:grid-cols-3 grid-cols-2',
|
|
|
|
7: 'xl:grid-cols-7 lg:grid-cols-5 md:grid-cols-4 grid-cols-2',
|
|
|
|
8: 'xl:grid-cols-8 lg:grid-cols-6 md:grid-cols-4 grid-cols-2',
|
|
|
|
};
|
|
|
|
|
|
|
|
const activeGridClass = gridClasses[maxCols] || gridClasses[5];
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<div className={`grid gap-4 ${activeGridClass}`}>
|
|
|
|
{cards.map((card) => (
|
|
|
|
<div
|
|
|
|
key={card.id}
|
|
|
|
onClick={() => setActiveCard(card)}
|
|
|
|
className="group relative cursor-pointer overflow-hidden rounded-xl shadow-md aspect-square bg-gray-200 transition-all duration-300 hover:shadow-xl hover:-translate-y-1"
|
|
|
|
>
|
|
|
|
{card.imageUrl ? (
|
|
|
|
<img src={card.imageUrl} alt={card.title} className="absolute inset-0 w-full h-full object-cover" />
|
|
|
|
) : (
|
|
|
|
<div className="absolute inset-0 flex items-center justify-center bg-gradient-to-br from-blue-100 to-gray-200 text-gray-400">No Image</div>
|
|
|
|
)}
|
|
|
|
<div className="absolute inset-0 bg-gradient-to-t from-black/90 via-black/30 to-transparent flex flex-col justify-end p-5 text-white">
|
|
|
|
<h3 className="text-xl font-bold drop-shadow-md">{card.title}</h3>
|
|
|
|
<p className="text-sm opacity-0 group-hover:opacity-100 transition-opacity duration-300 line-clamp-2 mt-1 text-gray-200 drop-shadow">
|
|
|
|
{card.shortDescription}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
{/* Improved Modal Pop-up */}
|
|
|
|
{activeCard && (
|
|
|
|
<div
|
|
|
|
className="fixed inset-0 z-50 flex items-center justify-center bg-black/70 backdrop-blur-md p-4 transition-opacity"
|
|
|
|
onClick={() => setActiveCard(null)} // Click outside to close
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className="bg-white rounded-2xl w-full max-w-2xl max-h-[90vh] overflow-y-auto shadow-2xl animate-in fade-in zoom-in-95 duration-200"
|
|
|
|
onClick={(e) => e.stopPropagation()} // Prevent clicks inside modal from closing it
|
|
|
|
>
|
|
|
|
<div className="relative h-72 w-full bg-gray-100">
|
|
|
|
{activeCard.imageUrl && (
|
|
|
|
<img src={activeCard.imageUrl} className="w-full h-full object-cover rounded-t-2xl" alt="" />
|
|
|
|
)}
|
|
|
|
{/* Improved Close Button */}
|
|
|
|
<button
|
|
|
|
onClick={() => setActiveCard(null)}
|
|
|
|
className="absolute top-4 right-4 bg-black/60 text-white w-10 h-10 flex items-center justify-center rounded-full hover:bg-black hover:scale-110 transition-all shadow-lg"
|
|
|
|
title="Close"
|
|
|
|
>
|
|
|
|
✕
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
<div className="p-8">
|
|
|
|
<div className="text-xs text-blue-600 font-bold tracking-wider uppercase mb-2">{activeCard.cardType.replace('_', ' ')}</div>
|
|
|
|
<h2 className="text-3xl font-bold mb-4 text-gray-900">{activeCard.title}</h2>
|
|
|
|
{activeCard.fullContent ? (
|
|
|
|
<div className="prose max-w-none text-gray-700" dangerouslySetInnerHTML={{ __html: activeCard.fullContent }} />
|
|
|
|
) : (
|
|
|
|
<p className="text-gray-500 italic text-lg">{activeCard.shortDescription}</p>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
} |
|
|
|
'use client'; |
|
|
|
import { useState, useEffect, useCallback, useRef } from 'react'; |
|
|
|
import { Card, MediaItem } from '@/types'; |
|
|
|
|
|
|
|
const isVideoUrl = (url: string) => /\.(mp4|webm|mov|m4v|ogv)(\?|$)/i.test(url); |
|
|
|
|
|
|
|
function MediaCarousel({ |
|
|
|
items, |
|
|
|
onMediaClick, |
|
|
|
}: { |
|
|
|
items: MediaItem[]; |
|
|
|
onMediaClick?: (index: number) => void; |
|
|
|
}) { |
|
|
|
const [current, setCurrent] = useState(0); |
|
|
|
const touchStartX = useRef<number | null>(null); |
|
|
|
const videoRefs = useRef<Record<number, HTMLVideoElement | null>>({}); |
|
|
|
|
|
|
|
const prev = useCallback(() => setCurrent(i => (i - 1 + items.length) % items.length), [items.length]); |
|
|
|
const next = useCallback(() => setCurrent(i => (i + 1) % items.length), [items.length]); |
|
|
|
|
|
|
|
useEffect(() => { |
|
|
|
const onKey = (e: KeyboardEvent) => { |
|
|
|
if (e.key === 'ArrowLeft') prev(); |
|
|
|
if (e.key === 'ArrowRight') next(); |
|
|
|
}; |
|
|
|
window.addEventListener('keydown', onKey); |
|
|
|
return () => window.removeEventListener('keydown', onKey); |
|
|
|
}, [prev, next]); |
|
|
|
|
|
|
|
useEffect(() => { setCurrent(0); }, [items]); |
|
|
|
|
|
|
|
// Pause non-current videos; autoplay current if flagged |
|
|
|
useEffect(() => { |
|
|
|
Object.entries(videoRefs.current).forEach(([key, vid]) => { |
|
|
|
if (!vid) return; |
|
|
|
const idx = parseInt(key, 10); |
|
|
|
if (idx !== current) { vid.pause(); return; } |
|
|
|
const item = items[idx]; |
|
|
|
if (item && isVideoUrl(item.url) && item.autoplay) { |
|
|
|
vid.muted = true; |
|
|
|
vid.play().catch(() => {}); |
|
|
|
} |
|
|
|
}); |
|
|
|
}, [current, items]); |
|
|
|
|
|
|
|
const onTouchStart = (e: React.TouchEvent) => { touchStartX.current = e.touches[0].clientX; }; |
|
|
|
const onTouchEnd = (e: React.TouchEvent) => { |
|
|
|
if (touchStartX.current === null) return; |
|
|
|
const delta = e.changedTouches[0].clientX - touchStartX.current; |
|
|
|
if (Math.abs(delta) > 50) delta < 0 ? next() : prev(); |
|
|
|
touchStartX.current = null; |
|
|
|
}; |
|
|
|
|
|
|
|
if (items.length === 0) { |
|
|
|
return <div className="h-72 w-full bg-gray-100 flex items-center justify-center text-gray-400 rounded-t-2xl">No Image</div>; |
|
|
|
} |
|
|
|
|
|
|
|
return ( |
|
|
|
<div |
|
|
|
className="relative h-72 w-full bg-black overflow-hidden rounded-t-2xl select-none" |
|
|
|
onTouchStart={onTouchStart} |
|
|
|
onTouchEnd={onTouchEnd} |
|
|
|
> |
|
|
|
{items.map((item, i) => { |
|
|
|
const isActive = i === current; |
|
|
|
const video = isVideoUrl(item.url); |
|
|
|
return ( |
|
|
|
<div |
|
|
|
key={item.url + i} |
|
|
|
className={`absolute inset-0 transition-opacity duration-300 ${isActive ? 'opacity-100' : 'opacity-0 pointer-events-none'}`} |
|
|
|
> |
|
|
|
{video ? ( |
|
|
|
<> |
|
|
|
<video |
|
|
|
ref={el => { videoRefs.current[i] = el; }} |
|
|
|
src={item.url} |
|
|
|
className="w-full h-full object-contain bg-black" |
|
|
|
controls |
|
|
|
controlsList="nofullscreen" |
|
|
|
disablePictureInPicture |
|
|
|
playsInline |
|
|
|
muted={!!item.autoplay} |
|
|
|
preload="metadata" |
|
|
|
/> |
|
|
|
<button |
|
|
|
onClick={(e) => { e.stopPropagation(); onMediaClick?.(i); }} |
|
|
|
className="absolute top-3 left-3 bg-black/60 hover:bg-black/80 text-white w-9 h-9 rounded-full flex items-center justify-center transition-colors shadow-lg z-10" |
|
|
|
title="Expand fullscreen" |
|
|
|
aria-label="Expand fullscreen" |
|
|
|
> |
|
|
|
<svg viewBox="0 0 24 24" className="w-4 h-4" fill="none" stroke="currentColor" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round"> |
|
|
|
<path d="M4 9V4h5M20 9V4h-5M4 15v5h5M20 15v5h-5" /> |
|
|
|
</svg> |
|
|
|
</button> |
|
|
|
</> |
|
|
|
) : ( |
|
|
|
<img |
|
|
|
src={item.url} |
|
|
|
alt="" |
|
|
|
className="w-full h-full object-cover cursor-zoom-in" |
|
|
|
onClick={() => onMediaClick?.(i)} |
|
|
|
title="Click to view fullscreen" |
|
|
|
/> |
|
|
|
)} |
|
|
|
</div> |
|
|
|
); |
|
|
|
})} |
|
|
|
|
|
|
|
{items.length > 1 && ( |
|
|
|
<> |
|
|
|
<button |
|
|
|
onClick={(e) => { e.stopPropagation(); prev(); }} |
|
|
|
className="absolute left-3 top-1/2 -translate-y-1/2 bg-black/50 hover:bg-black/80 text-white w-9 h-9 rounded-full flex items-center justify-center transition-colors shadow-lg z-10" |
|
|
|
aria-label="Previous" |
|
|
|
>‹</button> |
|
|
|
<button |
|
|
|
onClick={(e) => { e.stopPropagation(); next(); }} |
|
|
|
className="absolute right-3 top-1/2 -translate-y-1/2 bg-black/50 hover:bg-black/80 text-white w-9 h-9 rounded-full flex items-center justify-center transition-colors shadow-lg z-10" |
|
|
|
aria-label="Next" |
|
|
|
>›</button> |
|
|
|
|
|
|
|
<div className="absolute bottom-3 left-1/2 -translate-x-1/2 flex items-center gap-1.5 z-10"> |
|
|
|
{items.map((_, i) => ( |
|
|
|
<button |
|
|
|
key={i} |
|
|
|
onClick={(e) => { e.stopPropagation(); setCurrent(i); }} |
|
|
|
className={`rounded-full transition-all duration-200 ${i === current ? 'w-5 h-2 bg-white' : 'w-2 h-2 bg-white/50 hover:bg-white/80'}`} |
|
|
|
aria-label={`Go to slide ${i + 1}`} |
|
|
|
/> |
|
|
|
))} |
|
|
|
</div> |
|
|
|
|
|
|
|
<div className="absolute top-3 right-3 bg-black/50 text-white text-xs font-semibold px-2 py-0.5 rounded-full z-10"> |
|
|
|
{current + 1} / {items.length} |
|
|
|
</div> |
|
|
|
</> |
|
|
|
)} |
|
|
|
</div> |
|
|
|
); |
|
|
|
} |
|
|
|
|
|
|
|
function FullscreenViewer({ |
|
|
|
items, |
|
|
|
startIndex, |
|
|
|
onClose, |
|
|
|
}: { |
|
|
|
items: MediaItem[]; |
|
|
|
startIndex: number; |
|
|
|
onClose: () => void; |
|
|
|
}) { |
|
|
|
const [current, setCurrent] = useState(startIndex); |
|
|
|
const touchStartX = useRef<number | null>(null); |
|
|
|
|
|
|
|
const prev = useCallback(() => setCurrent(i => (i - 1 + items.length) % items.length), [items.length]); |
|
|
|
const next = useCallback(() => setCurrent(i => (i + 1) % items.length), [items.length]); |
|
|
|
|
|
|
|
useEffect(() => { |
|
|
|
const onKey = (e: KeyboardEvent) => { |
|
|
|
if (e.key === 'Escape') onClose(); |
|
|
|
else if (e.key === 'ArrowLeft') prev(); |
|
|
|
else if (e.key === 'ArrowRight') next(); |
|
|
|
}; |
|
|
|
window.addEventListener('keydown', onKey); |
|
|
|
return () => window.removeEventListener('keydown', onKey); |
|
|
|
}, [prev, next, onClose]); |
|
|
|
|
|
|
|
const onTouchStart = (e: React.TouchEvent) => { touchStartX.current = e.touches[0].clientX; }; |
|
|
|
const onTouchEnd = (e: React.TouchEvent) => { |
|
|
|
if (touchStartX.current === null) return; |
|
|
|
const delta = e.changedTouches[0].clientX - touchStartX.current; |
|
|
|
if (Math.abs(delta) > 50) delta < 0 ? next() : prev(); |
|
|
|
touchStartX.current = null; |
|
|
|
}; |
|
|
|
|
|
|
|
return ( |
|
|
|
<div |
|
|
|
className="fixed inset-0 z-[100] bg-black/95 flex items-center justify-center select-none animate-in fade-in duration-200" |
|
|
|
onTouchStart={onTouchStart} |
|
|
|
onTouchEnd={onTouchEnd} |
|
|
|
> |
|
|
|
{/* Media — full resolution, contained */} |
|
|
|
{items.map((item, i) => { |
|
|
|
const isActive = i === current; |
|
|
|
const video = isVideoUrl(item.url); |
|
|
|
return ( |
|
|
|
<div |
|
|
|
key={item.url + i} |
|
|
|
className={`absolute inset-0 flex items-center justify-center p-4 pb-28 transition-opacity duration-200 ${isActive ? 'opacity-100' : 'opacity-0 pointer-events-none'}`} |
|
|
|
> |
|
|
|
{video ? ( |
|
|
|
<video |
|
|
|
src={item.url} |
|
|
|
className="max-w-full max-h-full" |
|
|
|
controls |
|
|
|
controlsList="nofullscreen" |
|
|
|
disablePictureInPicture |
|
|
|
playsInline |
|
|
|
autoPlay={!!item.autoplay} |
|
|
|
muted={!!item.autoplay} |
|
|
|
/> |
|
|
|
) : ( |
|
|
|
<img |
|
|
|
src={item.url} |
|
|
|
alt="" |
|
|
|
className="max-w-full max-h-full object-contain" |
|
|
|
/> |
|
|
|
)} |
|
|
|
</div> |
|
|
|
); |
|
|
|
})} |
|
|
|
|
|
|
|
{/* Counter — top center */} |
|
|
|
{items.length > 1 && ( |
|
|
|
<div className="absolute top-4 left-1/2 -translate-x-1/2 bg-black/60 text-white text-sm font-semibold px-3 py-1 rounded-full z-20"> |
|
|
|
{current + 1} / {items.length} |
|
|
|
</div> |
|
|
|
)} |
|
|
|
|
|
|
|
{/* Side arrows */} |
|
|
|
{items.length > 1 && ( |
|
|
|
<> |
|
|
|
<button |
|
|
|
onClick={(e) => { e.stopPropagation(); prev(); }} |
|
|
|
className="absolute left-4 top-1/2 -translate-y-1/2 bg-black/50 hover:bg-black/80 text-white w-12 h-12 rounded-full flex items-center justify-center text-2xl shadow-lg z-20" |
|
|
|
aria-label="Previous" |
|
|
|
>‹</button> |
|
|
|
<button |
|
|
|
onClick={(e) => { e.stopPropagation(); next(); }} |
|
|
|
className="absolute right-4 top-1/2 -translate-y-1/2 bg-black/50 hover:bg-black/80 text-white w-12 h-12 rounded-full flex items-center justify-center text-2xl shadow-lg z-20" |
|
|
|
aria-label="Next" |
|
|
|
>›</button> |
|
|
|
</> |
|
|
|
)} |
|
|
|
|
|
|
|
{/* Close button — bottom center, ABOVE the dots */} |
|
|
|
<button |
|
|
|
onClick={onClose} |
|
|
|
className="absolute bottom-12 left-1/2 -translate-x-1/2 bg-white/90 hover:bg-white text-black px-6 py-2.5 rounded-full font-semibold shadow-2xl flex items-center gap-2 z-20 transition-transform hover:scale-105" |
|
|
|
aria-label="Close fullscreen" |
|
|
|
> |
|
|
|
<span className="text-lg leading-none">✕</span> |
|
|
|
<span>Close</span> |
|
|
|
</button> |
|
|
|
|
|
|
|
{/* Dots — at the very bottom */} |
|
|
|
{items.length > 1 && ( |
|
|
|
<div className="absolute bottom-4 left-1/2 -translate-x-1/2 flex items-center gap-1.5 z-20"> |
|
|
|
{items.map((_, i) => ( |
|
|
|
<button |
|
|
|
key={i} |
|
|
|
onClick={(e) => { e.stopPropagation(); setCurrent(i); }} |
|
|
|
className={`rounded-full transition-all duration-200 ${i === current ? 'w-6 h-2 bg-white' : 'w-2 h-2 bg-white/50 hover:bg-white/80'}`} |
|
|
|
aria-label={`Go to slide ${i + 1}`} |
|
|
|
/> |
|
|
|
))} |
|
|
|
</div> |
|
|
|
)} |
|
|
|
</div> |
|
|
|
); |
|
|
|
} |
|
|
|
|
|
|
|
export default function PublicGrid({ cards, maxCols = 5 }: { cards: Card[], maxCols?: number }) { |
|
|
|
const [activeCard, setActiveCard] = useState<Card | null>(null); |
|
|
|
const [fullscreenIndex, setFullscreenIndex] = useState<number | null>(null); |
|
|
|
|
|
|
|
useEffect(() => { |
|
|
|
if (activeCard || fullscreenIndex !== null) document.body.style.overflow = 'hidden'; |
|
|
|
else document.body.style.overflow = 'unset'; |
|
|
|
return () => { document.body.style.overflow = 'unset'; }; |
|
|
|
}, [activeCard, fullscreenIndex]); |
|
|
|
|
|
|
|
const gridClasses: Record<number, string> = { |
|
|
|
3: 'xl:grid-cols-3 lg:grid-cols-3 md:grid-cols-2 grid-cols-1', |
|
|
|
4: 'xl:grid-cols-4 lg:grid-cols-3 md:grid-cols-2 grid-cols-1', |
|
|
|
5: 'xl:grid-cols-5 lg:grid-cols-4 md:grid-cols-3 grid-cols-1', |
|
|
|
6: 'xl:grid-cols-6 lg:grid-cols-4 md:grid-cols-3 grid-cols-2', |
|
|
|
7: 'xl:grid-cols-7 lg:grid-cols-5 md:grid-cols-4 grid-cols-2', |
|
|
|
8: 'xl:grid-cols-8 lg:grid-cols-6 md:grid-cols-4 grid-cols-2', |
|
|
|
}; |
|
|
|
|
|
|
|
const activeGridClass = gridClasses[maxCols] || gridClasses[5]; |
|
|
|
|
|
|
|
const carouselItems: MediaItem[] = activeCard |
|
|
|
? [ |
|
|
|
...(activeCard.imageUrl ? [{ url: activeCard.imageUrl }] : []), |
|
|
|
...(activeCard.extraMedia || []), |
|
|
|
] |
|
|
|
: []; |
|
|
|
|
|
|
|
return ( |
|
|
|
<> |
|
|
|
<div className={`grid gap-4 ${activeGridClass}`}> |
|
|
|
{cards.map((card) => { |
|
|
|
const galleryCount = (card.extraMedia?.length || 0) + (card.imageUrl ? 1 : 0); |
|
|
|
return ( |
|
|
|
<div |
|
|
|
key={card.id} |
|
|
|
onClick={() => setActiveCard(card)} |
|
|
|
className="group relative cursor-pointer overflow-hidden rounded-xl shadow-md aspect-square bg-gray-200 transition-all duration-300 hover:shadow-xl hover:-translate-y-1" |
|
|
|
> |
|
|
|
{card.imageUrl ? ( |
|
|
|
<img src={card.imageUrl} alt={card.title} className="absolute inset-0 w-full h-full object-cover" /> |
|
|
|
) : ( |
|
|
|
<div className="absolute inset-0 flex items-center justify-center bg-gradient-to-br from-blue-100 to-gray-200 text-gray-400">No Image</div> |
|
|
|
)} |
|
|
|
{galleryCount > 1 && ( |
|
|
|
<div className="absolute top-2 right-2 bg-black/60 text-white text-xs font-semibold px-1.5 py-0.5 rounded-full flex items-center gap-1"> |
|
|
|
<span>⊞</span> |
|
|
|
<span>{galleryCount}</span> |
|
|
|
</div> |
|
|
|
)} |
|
|
|
<div className="absolute inset-0 bg-gradient-to-t from-black/90 via-black/30 to-transparent flex flex-col justify-end p-5 text-white"> |
|
|
|
<h3 className="text-xl font-bold drop-shadow-md">{card.title}</h3> |
|
|
|
<p className="text-sm opacity-0 group-hover:opacity-100 transition-opacity duration-300 line-clamp-2 mt-1 text-gray-200 drop-shadow"> |
|
|
|
{card.shortDescription} |
|
|
|
</p> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
); |
|
|
|
})} |
|
|
|
</div> |
|
|
|
|
|
|
|
{activeCard && ( |
|
|
|
<div |
|
|
|
className="fixed inset-0 z-50 flex items-center justify-center bg-black/70 backdrop-blur-md p-4" |
|
|
|
onClick={() => setActiveCard(null)} |
|
|
|
> |
|
|
|
<div |
|
|
|
className="bg-white rounded-2xl w-full max-w-2xl max-h-[90vh] overflow-y-auto shadow-2xl animate-in fade-in zoom-in-95 duration-200" |
|
|
|
onClick={(e) => e.stopPropagation()} |
|
|
|
> |
|
|
|
<div className="relative"> |
|
|
|
<MediaCarousel |
|
|
|
items={carouselItems} |
|
|
|
onMediaClick={(i) => setFullscreenIndex(i)} |
|
|
|
/> |
|
|
|
<button |
|
|
|
onClick={() => setActiveCard(null)} |
|
|
|
className="absolute top-4 right-4 bg-black/60 text-white w-10 h-10 flex items-center justify-center rounded-full hover:bg-black hover:scale-110 transition-all shadow-lg z-20" |
|
|
|
title="Close" |
|
|
|
>✕</button> |
|
|
|
</div> |
|
|
|
<div className="p-8"> |
|
|
|
<div className="text-xs text-blue-600 font-bold tracking-wider uppercase mb-2">{activeCard.cardType.replace('_', ' ')}</div> |
|
|
|
<h2 className="text-3xl font-bold mb-4 text-gray-900">{activeCard.title}</h2> |
|
|
|
{activeCard.fullContent ? ( |
|
|
|
<div className="prose max-w-none text-gray-700" dangerouslySetInnerHTML={{ __html: activeCard.fullContent }} /> |
|
|
|
) : ( |
|
|
|
<p className="text-gray-500 italic text-lg">{activeCard.shortDescription}</p> |
|
|
|
)} |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
)} |
|
|
|
|
|
|
|
{fullscreenIndex !== null && activeCard && ( |
|
|
|
<FullscreenViewer |
|
|
|
items={carouselItems} |
|
|
|
startIndex={fullscreenIndex} |
|
|
|
onClose={() => setFullscreenIndex(null)} |
|
|
|
/> |
|
|
|
)} |
|
|
|
</> |
|
|
|
); |
|
|
|
} |