369 lines
10 KiB
JavaScript
369 lines
10 KiB
JavaScript
/**
|
||
* Image Gallery Component
|
||
* Grid of images that open in a fullscreen modal on click
|
||
*/
|
||
class ImageGallery extends HTMLElement {
|
||
constructor() {
|
||
super();
|
||
this.attachShadow({ mode: "open" });
|
||
|
||
// Zoom and pan state
|
||
this.currentZoom = 1;
|
||
this.translateX = 0;
|
||
this.translateY = 0;
|
||
this.isDragging = false;
|
||
this.startX = 0;
|
||
this.startY = 0;
|
||
this.lastTranslateX = 0;
|
||
this.lastTranslateY = 0;
|
||
}
|
||
|
||
connectedCallback() {
|
||
this.render();
|
||
this.setupEventListeners();
|
||
}
|
||
|
||
get images() {
|
||
const imagesAttr = this.getAttribute("images");
|
||
return imagesAttr ? imagesAttr.split(",").map((img) => img.trim()) : [];
|
||
}
|
||
|
||
setupEventListeners() {
|
||
const imageItems = this.shadowRoot.querySelectorAll(".gallery-image");
|
||
imageItems.forEach((img, index) => {
|
||
img.addEventListener("click", () => {
|
||
this.openModal(index);
|
||
});
|
||
});
|
||
|
||
// Modal controls
|
||
const closeBtn = this.shadowRoot.querySelector(".modal-close");
|
||
const modal = this.shadowRoot.querySelector(".modal-overlay");
|
||
|
||
if (closeBtn) {
|
||
closeBtn.addEventListener("click", () => this.closeModal());
|
||
}
|
||
|
||
if (modal) {
|
||
modal.addEventListener("click", (e) => {
|
||
if (e.target === modal) {
|
||
this.closeModal();
|
||
}
|
||
});
|
||
}
|
||
|
||
// Keyboard support
|
||
document.addEventListener("keydown", (e) => {
|
||
if (e.key === "Escape") {
|
||
this.closeModal();
|
||
}
|
||
});
|
||
|
||
// Zoom controls
|
||
const zoomIn = this.shadowRoot.querySelector(".zoom-in");
|
||
const zoomOut = this.shadowRoot.querySelector(".zoom-out");
|
||
const resetZoom = this.shadowRoot.querySelector(".zoom-reset");
|
||
|
||
if (zoomIn) {
|
||
zoomIn.addEventListener("click", () => this.zoom(0.25));
|
||
}
|
||
if (zoomOut) {
|
||
zoomOut.addEventListener("click", () => this.zoom(-0.25));
|
||
}
|
||
if (resetZoom) {
|
||
resetZoom.addEventListener("click", () => this.resetZoom());
|
||
}
|
||
|
||
// Drag/pan functionality
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
// Mouse events
|
||
modalImage.addEventListener("mousedown", (e) => this.startDrag(e));
|
||
modalImage.addEventListener("mousemove", (e) => this.drag(e));
|
||
modalImage.addEventListener("mouseup", () => this.endDrag());
|
||
modalImage.addEventListener("mouseleave", () => this.endDrag());
|
||
|
||
// Touch events for mobile
|
||
modalImage.addEventListener("touchstart", (e) => this.startDrag(e), { passive: false });
|
||
modalImage.addEventListener("touchmove", (e) => this.drag(e), { passive: false });
|
||
modalImage.addEventListener("touchend", () => this.endDrag());
|
||
|
||
// Prevent default drag behavior
|
||
modalImage.addEventListener("dragstart", (e) => e.preventDefault());
|
||
}
|
||
}
|
||
|
||
startDrag(e) {
|
||
// Only enable drag when zoomed in
|
||
if (this.currentZoom <= 1) return;
|
||
|
||
this.isDragging = true;
|
||
|
||
if (e.type === "touchstart") {
|
||
this.startX = e.touches[0].clientX - this.translateX;
|
||
this.startY = e.touches[0].clientY - this.translateY;
|
||
} else {
|
||
this.startX = e.clientX - this.translateX;
|
||
this.startY = e.clientY - this.translateY;
|
||
}
|
||
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
modalImage.style.cursor = "grabbing";
|
||
modalImage.style.transition = "none";
|
||
}
|
||
}
|
||
|
||
drag(e) {
|
||
if (!this.isDragging) return;
|
||
|
||
e.preventDefault();
|
||
|
||
let clientX, clientY;
|
||
if (e.type === "touchmove") {
|
||
clientX = e.touches[0].clientX;
|
||
clientY = e.touches[0].clientY;
|
||
} else {
|
||
clientX = e.clientX;
|
||
clientY = e.clientY;
|
||
}
|
||
|
||
this.translateX = clientX - this.startX;
|
||
this.translateY = clientY - this.startY;
|
||
|
||
this.updateImageTransform();
|
||
}
|
||
|
||
endDrag() {
|
||
if (!this.isDragging) return;
|
||
|
||
this.isDragging = false;
|
||
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
modalImage.style.cursor = this.currentZoom > 1 ? "grab" : "default";
|
||
modalImage.style.transition = "transform 0.2s ease";
|
||
}
|
||
}
|
||
|
||
updateImageTransform() {
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
modalImage.style.transform = `scale(${this.currentZoom}) translate(${this.translateX / this.currentZoom}px, ${this.translateY / this.currentZoom}px)`;
|
||
}
|
||
}
|
||
|
||
openModal(index) {
|
||
const modal = this.shadowRoot.querySelector(".modal-overlay");
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
|
||
if (modal && modalImage && this.images[index]) {
|
||
modalImage.src = this.images[index];
|
||
|
||
// Reset zoom and pan
|
||
this.currentZoom = 1;
|
||
this.translateX = 0;
|
||
this.translateY = 0;
|
||
this.updateImageTransform();
|
||
|
||
modalImage.style.cursor = "default";
|
||
modal.classList.add("open");
|
||
document.body.style.overflow = "hidden";
|
||
}
|
||
}
|
||
|
||
closeModal() {
|
||
const modal = this.shadowRoot.querySelector(".modal-overlay");
|
||
if (modal) {
|
||
modal.classList.remove("open");
|
||
document.body.style.overflow = "";
|
||
}
|
||
}
|
||
|
||
zoom(delta) {
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
const prevZoom = this.currentZoom;
|
||
this.currentZoom = Math.max(1, Math.min(4, this.currentZoom + delta));
|
||
|
||
// If zooming out to 1x, reset position
|
||
if (this.currentZoom === 1) {
|
||
this.translateX = 0;
|
||
this.translateY = 0;
|
||
}
|
||
|
||
this.updateImageTransform();
|
||
modalImage.style.cursor = this.currentZoom > 1 ? "grab" : "default";
|
||
}
|
||
}
|
||
|
||
resetZoom() {
|
||
const modalImage = this.shadowRoot.querySelector(".modal-image");
|
||
if (modalImage) {
|
||
this.currentZoom = 1;
|
||
this.translateX = 0;
|
||
this.translateY = 0;
|
||
this.updateImageTransform();
|
||
modalImage.style.cursor = "default";
|
||
}
|
||
}
|
||
|
||
render() {
|
||
const imagesHtml = this.images
|
||
.map(
|
||
(img, index) => `
|
||
<button class="gallery-item" type="button" aria-label="View image ${index + 1}">
|
||
<img src="${img}" alt="Book preview ${index + 1}" class="gallery-image" />
|
||
</button>
|
||
`
|
||
)
|
||
.join("");
|
||
|
||
this.shadowRoot.innerHTML = `
|
||
<style>
|
||
:host {
|
||
display: block;
|
||
}
|
||
|
||
.gallery-grid {
|
||
display: grid;
|
||
grid-template-columns: repeat(3, 1fr);
|
||
gap: var(--spacing-sm, 0.5rem);
|
||
}
|
||
|
||
.gallery-item {
|
||
aspect-ratio: 3/4;
|
||
overflow: hidden;
|
||
border-radius: var(--radius-sm, 0.25rem);
|
||
background: none;
|
||
border: none;
|
||
padding: 0;
|
||
cursor: pointer;
|
||
transition: opacity var(--transition-fast, 150ms ease);
|
||
}
|
||
|
||
.gallery-item:hover {
|
||
opacity: 0.8;
|
||
}
|
||
|
||
.gallery-image {
|
||
width: 100%;
|
||
height: 100%;
|
||
object-fit: cover;
|
||
}
|
||
|
||
/* Modal Styles */
|
||
.modal-overlay {
|
||
display: none;
|
||
position: fixed;
|
||
top: 0;
|
||
left: 0;
|
||
right: 0;
|
||
bottom: 0;
|
||
background-color: rgba(0, 0, 0, 0.95);
|
||
z-index: 1000;
|
||
flex-direction: column;
|
||
}
|
||
|
||
.modal-overlay.open {
|
||
display: flex;
|
||
}
|
||
|
||
.modal-header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
padding: var(--spacing-md, 1rem);
|
||
background-color: rgba(0, 0, 0, 0.5);
|
||
}
|
||
|
||
.modal-close {
|
||
width: 40px;
|
||
height: 40px;
|
||
background: none;
|
||
border: none;
|
||
cursor: pointer;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
margin-left: auto;
|
||
}
|
||
|
||
.modal-close svg {
|
||
width: 32px;
|
||
height: 32px;
|
||
color: #ffffff;
|
||
}
|
||
|
||
.zoom-controls {
|
||
display: flex;
|
||
gap: var(--spacing-sm, 0.5rem);
|
||
}
|
||
|
||
.zoom-btn {
|
||
width: 40px;
|
||
height: 40px;
|
||
background: rgba(255, 255, 255, 0.1);
|
||
border: 1px solid rgba(255, 255, 255, 0.3);
|
||
border-radius: var(--radius-sm, 0.25rem);
|
||
color: #ffffff;
|
||
cursor: pointer;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
font-size: 20px;
|
||
transition: background var(--transition-fast, 150ms ease);
|
||
}
|
||
|
||
.zoom-btn:hover {
|
||
background: rgba(255, 255, 255, 0.2);
|
||
}
|
||
|
||
.modal-content {
|
||
flex: 1;
|
||
overflow: auto;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
padding: var(--spacing-md, 1rem);
|
||
}
|
||
|
||
.modal-image {
|
||
max-width: 100%;
|
||
max-height: 100%;
|
||
object-fit: contain;
|
||
transition: transform 0.2s ease;
|
||
cursor: default;
|
||
user-select: none;
|
||
-webkit-user-drag: none;
|
||
}
|
||
</style>
|
||
|
||
<div class="gallery-grid">
|
||
${imagesHtml}
|
||
</div>
|
||
|
||
<div class="modal-overlay">
|
||
<div class="modal-header">
|
||
<div class="zoom-controls">
|
||
<button class="zoom-btn zoom-out" type="button" aria-label="Zoom out">−</button>
|
||
<button class="zoom-btn zoom-reset" type="button" aria-label="Reset zoom">⟲</button>
|
||
<button class="zoom-btn zoom-in" type="button" aria-label="Zoom in">+</button>
|
||
</div>
|
||
<button class="modal-close" type="button" aria-label="Close">
|
||
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||
<line x1="18" y1="6" x2="6" y2="18"></line>
|
||
<line x1="6" y1="6" x2="18" y2="18"></line>
|
||
</svg>
|
||
</button>
|
||
</div>
|
||
<div class="modal-content">
|
||
<img class="modal-image" src="" alt="Full size preview" />
|
||
</div>
|
||
</div>
|
||
`;
|
||
}
|
||
}
|
||
|
||
customElements.define("image-gallery", ImageGallery);
|