/** * Crop tool — drag-rect selection that lets the user cut down the * canvas to a smaller region. Supports Shift-lock aspect ratio and * click-inside-rect to reposition an existing crop without redrawing. * * Owns its own begin/drag/end handlers and reads/writes shared state. * The factory takes a small dependency bag for things still living in * galleryEditor.js — `composite` redraws the canvas, `showCropApply` * mounts the floating W×H + Apply panel after the user finishes * dragging. * * @param {{ * composite: () => void, * showCropApply: () => void, * }} deps */ import { state } from '../state.js'; import { canvasCoords } from '../canvas-coords.js'; import { drawCheckerboard } from '../checkerboard.js'; export function createCropTool({ composite, showCropApply }) { return { begin(e) { const coords = canvasCoords(e, state.mainCanvas); // Click inside an existing crop rect → switch to move-mode so // the user can reposition without redrawing. if (state.cropRect && coords.x >= state.cropRect.x && coords.x <= state.cropRect.x + state.cropRect.w && coords.y >= state.cropRect.y && coords.y <= state.cropRect.y + state.cropRect.h) { state.cropMoving = true; state.cropMoveStart = { x: coords.x, y: coords.y, rx: state.cropRect.x, ry: state.cropRect.y }; return; } state.cropping = true; state.cropStart = coords; state.cropEnd = { ...state.cropStart }; state.cropRect = null; state.cropAspectLock = null; // Tear down the size panel while the user is drawing a new rect. const old = state.container?.querySelector('.ge-crop-apply'); if (old) old.remove(); }, drag(e) { // Move-mode: drag the existing rect around the canvas. if (state.cropMoving && state.cropRect && state.cropMoveStart) { e.preventDefault(); const c = canvasCoords(e, state.mainCanvas); const dx = c.x - state.cropMoveStart.x; const dy = c.y - state.cropMoveStart.y; let nx = state.cropMoveStart.rx + dx; let ny = state.cropMoveStart.ry + dy; // Clamp to canvas bounds so the rect stays fully visible. nx = Math.max(0, Math.min(nx, state.mainCanvas.width - state.cropRect.w)); ny = Math.max(0, Math.min(ny, state.mainCanvas.height - state.cropRect.h)); state.cropRect = { ...state.cropRect, x: nx, y: ny }; composite(); return; } if (!state.cropping) return; e.preventDefault(); state.cropEnd = canvasCoords(e, state.mainCanvas); // Shift-held = lock aspect ratio. First Shift press during the // drag snapshots the current aspect; subsequent moves stay locked. // Releasing Shift resets so the user can re-lock at a new ratio. if (e.shiftKey) { const rawDx = state.cropEnd.x - state.cropStart.x; const rawDy = state.cropEnd.y - state.cropStart.y; if (state.cropAspectLock == null) { const rawW = Math.abs(rawDx) || 1; const rawH = Math.abs(rawDy) || 1; state.cropAspectLock = rawW / rawH; } const absDx = Math.abs(rawDx); const absDy = Math.abs(rawDy); // Whichever axis the user moved more (relative to the lock) is // the driver; scale the other to preserve aspect. let dx, dy; if (absDx >= absDy * state.cropAspectLock) { dx = rawDx; dy = Math.sign(rawDy || 1) * (absDx / state.cropAspectLock); } else { dy = rawDy; dx = Math.sign(rawDx || 1) * (absDy * state.cropAspectLock); } state.cropEnd = { x: state.cropStart.x + dx, y: state.cropStart.y + dy }; } else { state.cropAspectLock = null; } composite(); // Draw crop overlay. const x = Math.min(state.cropStart.x, state.cropEnd.x); const y = Math.min(state.cropStart.y, state.cropEnd.y); const w = Math.abs(state.cropEnd.x - state.cropStart.x); const h = Math.abs(state.cropEnd.y - state.cropStart.y); state.mainCtx.fillStyle = 'rgba(0,0,0,0.4)'; state.mainCtx.fillRect(0, 0, state.mainCanvas.width, state.mainCanvas.height); state.mainCtx.clearRect(x, y, w, h); // Redraw layers inside the crop rect (dim everything outside). state.mainCtx.save(); state.mainCtx.beginPath(); state.mainCtx.rect(x, y, w, h); state.mainCtx.clip(); drawCheckerboard(state.mainCtx, state.mainCanvas.width, state.mainCanvas.height); for (const layer of state.layers) { if (!layer.visible) continue; state.mainCtx.globalAlpha = layer.opacity; const off = state.layerOffsets.get(layer.id) || { x: 0, y: 0 }; state.mainCtx.drawImage(layer.canvas, off.x, off.y); } state.mainCtx.globalAlpha = 1; state.mainCtx.restore(); // Dashed border around the kept region. state.mainCtx.strokeStyle = '#fff'; state.mainCtx.lineWidth = 1; state.mainCtx.setLineDash([4, 4]); state.mainCtx.strokeRect(x, y, w, h); state.mainCtx.setLineDash([]); state.cropRect = { x, y, w, h }; }, end() { // Move-mode wrap-up: refresh the floating panel so Apply follows // the rect to its new spot. if (state.cropMoving) { state.cropMoving = false; state.cropMoveStart = null; if (state.cropRect) showCropApply(); return; } state.cropping = false; if (state.cropRect && state.cropRect.w > 5 && state.cropRect.h > 5) { showCropApply(); } }, }; }