aboutsummaryrefslogtreecommitdiff
path: root/src/components/ImageGalleryGrid.tsx
blob: fc651bf0f2a8fe92dcfb2589df469d13a1cf3301 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
import React, { useState, useEffect } from 'react';

interface Props {
  images: Array<{
    src: string;
    alt?: string;
    id?: string | number;
  }>;
}

export default function ImageGalleryGrid({ images }: Props) {
  const [selectedIndex, setSelectedIndex] = useState<number | null>(null);
  const closeModal = () => setSelectedIndex(null);
  const showPrev = (e: React.MouseEvent) => {
    e.stopPropagation();
    setSelectedIndex((prev) => (prev !== null && prev > 0 ? prev - 1 : prev));
  };
  const showNext = (e: React.MouseEvent) => {
    e.stopPropagation();
    setSelectedIndex((prev) =>
      prev !== null && prev < images.length - 1 ? prev + 1 : prev
    );
  };

  useEffect(() => {
    function handleKeyDown(event: KeyboardEvent) {
      if (selectedIndex === null) return;
      if (event.key === 'Escape') {
        closeModal;
      } else if (event.key === 'ArrowLeft') {
        setSelectedIndex((prev) => (prev && prev > 0 ? prev - 1 : prev));
      } else if (event.key === 'ArrowRight') {
        setSelectedIndex((prev) =>
          prev !== null && prev < images.length - 1 ? prev + 1 : prev
        );
      }
    }

    window.addEventListener('keydown', handleKeyDown);
    return () => window.removeEventListener('keydown', handleKeyDown);
  }, [selectedIndex, images.length]);

  return (
    <>
      <div className="grid gap-2 grid-cols-4 p-4">
        {images.map((image, idx) => (
          <div
            key={image.id ?? image.src}
            className="relative group overflow-hidden rounded-xl shadow-lg cursor-pointer aspect-square"
            onClick={() => setSelectedIndex(idx)}
          >
            <img
              src={image.src}
              alt={image.alt || 'Gallery image'}
              className="w-full h-full object-cover group-hover:scale-105 transition-transform duration-300"
            />
          </div>
        ))}
      </div>

      {/* Overlay */}
      {selectedIndex !== null && (
        <div
          className="fixed inset-0 bg-white/50 dark:bg-black/50 backdrop-blur-sm flex items-center justify-center z-50"
          onClick={closeModal}
          role="dialog"
          aria-modal="true"
        >
          {/* Prev arrow */}
          <button
            onClick={showPrev}
            className="absolute left-4 text-black dark:text-white text-4xl focus:outline-none cursor-pointer"
            disabled={selectedIndex === 0}
          >
            &#8592;
          </button>

          <img
            src={images[selectedIndex].src}
            alt={images[selectedIndex].alt || 'Enlarged gallery'}
            className="w-1/2 h-1/2 object-contain rounded-lg"
          />

          {/* Next arrow */}
          <button
            onClick={showNext}
            className="absolute right-4 text-black dark:text-white text-4xl focus:outline-none cursor-pointer"
            disabled={selectedIndex === images.length - 1}
          >
            &#8594;
          </button>
        </div>
      )}
    </>
  );
}