138 lines
5.5 KiB
JavaScript
138 lines
5.5 KiB
JavaScript
/**
|
||
* 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();
|
||
}
|
||
},
|
||
};
|
||
}
|