mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
refactor(ui): generalize stage event handlers
Create intermediary nanostores for values required by the event handlers. This allows the event handlers to be purely imperative, with no reactivity: instead of recreating/setting the handlers when a dependent piece of state changes, we use nanostores' imperative API to access dependent state. For example, some handlers depend on brush size. If we used the standard declarative `useSelector` API, we'd need to recreate the event handler callback each time the brush size changed. This can be costly. An intermediate `$brushSize` nanostore is set in a `useLayoutEffect()`, which responds to changes to the redux store. Then, in the event handler, we use the imperative API to access the brush size: `$brushSize.get()`. This change allows the event handler logic to be shared with the pending canvas v2, and also more easily tested. It's a noticeable perf improvement, too, especially when changing brush size.
This commit is contained in:
parent
1823e446ac
commit
3db69af220
@ -4,19 +4,38 @@ import { createSelector } from '@reduxjs/toolkit';
|
|||||||
import { logger } from 'app/logging/logger';
|
import { logger } from 'app/logging/logger';
|
||||||
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import { useMouseEvents } from 'features/controlLayers/hooks/mouseEventHooks';
|
|
||||||
import {
|
import {
|
||||||
|
$brushSize,
|
||||||
|
$brushSpacingPx,
|
||||||
|
$isDrawing,
|
||||||
|
$lastAddedPoint,
|
||||||
$lastCursorPos,
|
$lastCursorPos,
|
||||||
$lastMouseDownPos,
|
$lastMouseDownPos,
|
||||||
|
$selectedLayerId,
|
||||||
|
$selectedLayerType,
|
||||||
|
$shouldInvertBrushSizeScrollDirection,
|
||||||
$tool,
|
$tool,
|
||||||
|
BRUSH_SPACING_PCT,
|
||||||
|
brushSizeChanged,
|
||||||
isRegionalGuidanceLayer,
|
isRegionalGuidanceLayer,
|
||||||
layerBboxChanged,
|
layerBboxChanged,
|
||||||
layerTranslated,
|
layerTranslated,
|
||||||
|
MAX_BRUSH_SPACING_PX,
|
||||||
|
MIN_BRUSH_SPACING_PX,
|
||||||
|
rgLayerLineAdded,
|
||||||
|
rgLayerPointsAdded,
|
||||||
|
rgLayerRectAdded,
|
||||||
selectControlLayersSlice,
|
selectControlLayersSlice,
|
||||||
} from 'features/controlLayers/store/controlLayersSlice';
|
} from 'features/controlLayers/store/controlLayersSlice';
|
||||||
import { debouncedRenderers, renderers as normalRenderers } from 'features/controlLayers/util/renderers';
|
import type { AddLineArg, AddPointToLineArg, AddRectArg } from 'features/controlLayers/store/types';
|
||||||
|
import {
|
||||||
|
debouncedRenderers,
|
||||||
|
renderers as normalRenderers,
|
||||||
|
setStageEventHandlers,
|
||||||
|
} from 'features/controlLayers/util/renderers';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
import type { IRect } from 'konva/lib/types';
|
import type { IRect } from 'konva/lib/types';
|
||||||
|
import { clamp } from 'lodash-es';
|
||||||
import { memo, useCallback, useLayoutEffect, useMemo, useState } from 'react';
|
import { memo, useCallback, useLayoutEffect, useMemo, useState } from 'react';
|
||||||
import { getImageDTO } from 'services/api/endpoints/images';
|
import { getImageDTO } from 'services/api/endpoints/images';
|
||||||
import { useDevicePixelRatio } from 'use-device-pixel-ratio';
|
import { useDevicePixelRatio } from 'use-device-pixel-ratio';
|
||||||
@ -48,7 +67,6 @@ const useStageRenderer = (
|
|||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const state = useAppSelector((s) => s.controlLayers.present);
|
const state = useAppSelector((s) => s.controlLayers.present);
|
||||||
const tool = useStore($tool);
|
const tool = useStore($tool);
|
||||||
const mouseEventHandlers = useMouseEvents();
|
|
||||||
const lastCursorPos = useStore($lastCursorPos);
|
const lastCursorPos = useStore($lastCursorPos);
|
||||||
const lastMouseDownPos = useStore($lastMouseDownPos);
|
const lastMouseDownPos = useStore($lastMouseDownPos);
|
||||||
const selectedLayerIdColor = useAppSelector(selectSelectedLayerColor);
|
const selectedLayerIdColor = useAppSelector(selectSelectedLayerColor);
|
||||||
@ -57,6 +75,26 @@ const useStageRenderer = (
|
|||||||
const layerCount = useMemo(() => state.layers.length, [state.layers]);
|
const layerCount = useMemo(() => state.layers.length, [state.layers]);
|
||||||
const renderers = useMemo(() => (asPreview ? debouncedRenderers : normalRenderers), [asPreview]);
|
const renderers = useMemo(() => (asPreview ? debouncedRenderers : normalRenderers), [asPreview]);
|
||||||
const dpr = useDevicePixelRatio({ round: false });
|
const dpr = useDevicePixelRatio({ round: false });
|
||||||
|
const shouldInvertBrushSizeScrollDirection = useAppSelector((s) => s.canvas.shouldInvertBrushSizeScrollDirection);
|
||||||
|
const brushSpacingPx = useMemo(
|
||||||
|
() => clamp(state.brushSize / BRUSH_SPACING_PCT, MIN_BRUSH_SPACING_PX, MAX_BRUSH_SPACING_PX),
|
||||||
|
[state.brushSize]
|
||||||
|
);
|
||||||
|
|
||||||
|
useLayoutEffect(() => {
|
||||||
|
$brushSize.set(state.brushSize);
|
||||||
|
$brushSpacingPx.set(brushSpacingPx);
|
||||||
|
$selectedLayerId.set(state.selectedLayerId);
|
||||||
|
$selectedLayerType.set(selectedLayerType);
|
||||||
|
$shouldInvertBrushSizeScrollDirection.set(shouldInvertBrushSizeScrollDirection);
|
||||||
|
}, [
|
||||||
|
brushSpacingPx,
|
||||||
|
selectedLayerIdColor,
|
||||||
|
selectedLayerType,
|
||||||
|
shouldInvertBrushSizeScrollDirection,
|
||||||
|
state.brushSize,
|
||||||
|
state.selectedLayerId,
|
||||||
|
]);
|
||||||
|
|
||||||
const onLayerPosChanged = useCallback(
|
const onLayerPosChanged = useCallback(
|
||||||
(layerId: string, x: number, y: number) => {
|
(layerId: string, x: number, y: number) => {
|
||||||
@ -72,6 +110,31 @@ const useStageRenderer = (
|
|||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const onRGLayerLineAdded = useCallback(
|
||||||
|
(arg: AddLineArg) => {
|
||||||
|
dispatch(rgLayerLineAdded(arg));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const onRGLayerPointAddedToLine = useCallback(
|
||||||
|
(arg: AddPointToLineArg) => {
|
||||||
|
dispatch(rgLayerPointsAdded(arg));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const onRGLayerRectAdded = useCallback(
|
||||||
|
(arg: AddRectArg) => {
|
||||||
|
dispatch(rgLayerRectAdded(arg));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
const onBrushSizeChanged = useCallback(
|
||||||
|
(size: number) => {
|
||||||
|
dispatch(brushSizeChanged(size));
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
log.trace('Initializing stage');
|
log.trace('Initializing stage');
|
||||||
if (!container) {
|
if (!container) {
|
||||||
@ -89,21 +152,29 @@ const useStageRenderer = (
|
|||||||
if (asPreview) {
|
if (asPreview) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
stage.on('mousedown', mouseEventHandlers.onMouseDown);
|
const cleanup = setStageEventHandlers({
|
||||||
stage.on('mouseup', mouseEventHandlers.onMouseUp);
|
stage,
|
||||||
stage.on('mousemove', mouseEventHandlers.onMouseMove);
|
$tool,
|
||||||
stage.on('mouseleave', mouseEventHandlers.onMouseLeave);
|
$isDrawing,
|
||||||
stage.on('wheel', mouseEventHandlers.onMouseWheel);
|
$lastMouseDownPos,
|
||||||
|
$lastCursorPos,
|
||||||
|
$lastAddedPoint,
|
||||||
|
$brushSize,
|
||||||
|
$brushSpacingPx,
|
||||||
|
$selectedLayerId,
|
||||||
|
$selectedLayerType,
|
||||||
|
$shouldInvertBrushSizeScrollDirection,
|
||||||
|
onRGLayerLineAdded,
|
||||||
|
onRGLayerPointAddedToLine,
|
||||||
|
onRGLayerRectAdded,
|
||||||
|
onBrushSizeChanged,
|
||||||
|
});
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
log.trace('Cleaning up stage listeners');
|
log.trace('Removing stage listeners');
|
||||||
stage.off('mousedown', mouseEventHandlers.onMouseDown);
|
cleanup();
|
||||||
stage.off('mouseup', mouseEventHandlers.onMouseUp);
|
|
||||||
stage.off('mousemove', mouseEventHandlers.onMouseMove);
|
|
||||||
stage.off('mouseleave', mouseEventHandlers.onMouseLeave);
|
|
||||||
stage.off('wheel', mouseEventHandlers.onMouseWheel);
|
|
||||||
};
|
};
|
||||||
}, [stage, asPreview, mouseEventHandlers]);
|
}, [asPreview, onBrushSizeChanged, onRGLayerLineAdded, onRGLayerPointAddedToLine, onRGLayerRectAdded, stage]);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
log.trace('Updating stage dimensions');
|
log.trace('Updating stage dimensions');
|
||||||
|
@ -1,233 +0,0 @@
|
|||||||
import { $ctrl, $meta } from '@invoke-ai/ui-library';
|
|
||||||
import { useStore } from '@nanostores/react';
|
|
||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
|
||||||
import { calculateNewBrushSize } from 'features/canvas/hooks/useCanvasZoom';
|
|
||||||
import {
|
|
||||||
$isDrawing,
|
|
||||||
$lastCursorPos,
|
|
||||||
$lastMouseDownPos,
|
|
||||||
$tool,
|
|
||||||
brushSizeChanged,
|
|
||||||
rgLayerLineAdded,
|
|
||||||
rgLayerPointsAdded,
|
|
||||||
rgLayerRectAdded,
|
|
||||||
} from 'features/controlLayers/store/controlLayersSlice';
|
|
||||||
import type Konva from 'konva';
|
|
||||||
import type { KonvaEventObject } from 'konva/lib/Node';
|
|
||||||
import type { Vector2d } from 'konva/lib/types';
|
|
||||||
import { clamp } from 'lodash-es';
|
|
||||||
import { useCallback, useMemo, useRef } from 'react';
|
|
||||||
|
|
||||||
const getIsFocused = (stage: Konva.Stage) => {
|
|
||||||
return stage.container().contains(document.activeElement);
|
|
||||||
};
|
|
||||||
const getIsMouseDown = (e: KonvaEventObject<MouseEvent>) => e.evt.buttons === 1;
|
|
||||||
|
|
||||||
const SNAP_PX = 10;
|
|
||||||
|
|
||||||
export const snapPosToStage = (pos: Vector2d, stage: Konva.Stage) => {
|
|
||||||
const snappedPos = { ...pos };
|
|
||||||
// Get the normalized threshold for snapping to the edge of the stage
|
|
||||||
const thresholdX = SNAP_PX / stage.scaleX();
|
|
||||||
const thresholdY = SNAP_PX / stage.scaleY();
|
|
||||||
const stageWidth = stage.width() / stage.scaleX();
|
|
||||||
const stageHeight = stage.height() / stage.scaleY();
|
|
||||||
// Snap to the edge of the stage if within threshold
|
|
||||||
if (pos.x - thresholdX < 0) {
|
|
||||||
snappedPos.x = 0;
|
|
||||||
} else if (pos.x + thresholdX > stageWidth) {
|
|
||||||
snappedPos.x = Math.floor(stageWidth);
|
|
||||||
}
|
|
||||||
if (pos.y - thresholdY < 0) {
|
|
||||||
snappedPos.y = 0;
|
|
||||||
} else if (pos.y + thresholdY > stageHeight) {
|
|
||||||
snappedPos.y = Math.floor(stageHeight);
|
|
||||||
}
|
|
||||||
return snappedPos;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getScaledFlooredCursorPosition = (stage: Konva.Stage) => {
|
|
||||||
const pointerPosition = stage.getPointerPosition();
|
|
||||||
const stageTransform = stage.getAbsoluteTransform().copy();
|
|
||||||
if (!pointerPosition) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const scaledCursorPosition = stageTransform.invert().point(pointerPosition);
|
|
||||||
return {
|
|
||||||
x: Math.floor(scaledCursorPosition.x),
|
|
||||||
y: Math.floor(scaledCursorPosition.y),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const syncCursorPos = (stage: Konva.Stage): Vector2d | null => {
|
|
||||||
const pos = getScaledFlooredCursorPosition(stage);
|
|
||||||
if (!pos) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
$lastCursorPos.set(pos);
|
|
||||||
return pos;
|
|
||||||
};
|
|
||||||
|
|
||||||
const BRUSH_SPACING_PCT = 10;
|
|
||||||
const MIN_BRUSH_SPACING_PX = 5;
|
|
||||||
const MAX_BRUSH_SPACING_PX = 15;
|
|
||||||
|
|
||||||
export const useMouseEvents = () => {
|
|
||||||
const dispatch = useAppDispatch();
|
|
||||||
const selectedLayerId = useAppSelector((s) => s.controlLayers.present.selectedLayerId);
|
|
||||||
const selectedLayerType = useAppSelector((s) => {
|
|
||||||
const selectedLayer = s.controlLayers.present.layers.find((l) => l.id === s.controlLayers.present.selectedLayerId);
|
|
||||||
if (!selectedLayer) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return selectedLayer.type;
|
|
||||||
});
|
|
||||||
const tool = useStore($tool);
|
|
||||||
const lastCursorPosRef = useRef<[number, number] | null>(null);
|
|
||||||
const shouldInvertBrushSizeScrollDirection = useAppSelector((s) => s.canvas.shouldInvertBrushSizeScrollDirection);
|
|
||||||
const brushSize = useAppSelector((s) => s.controlLayers.present.brushSize);
|
|
||||||
const brushSpacingPx = useMemo(
|
|
||||||
() => clamp(brushSize / BRUSH_SPACING_PCT, MIN_BRUSH_SPACING_PX, MAX_BRUSH_SPACING_PX),
|
|
||||||
[brushSize]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onMouseDown = useCallback(
|
|
||||||
(e: KonvaEventObject<MouseEvent>) => {
|
|
||||||
const stage = e.target.getStage();
|
|
||||||
if (!stage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pos = syncCursorPos(stage);
|
|
||||||
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (tool === 'brush' || tool === 'eraser') {
|
|
||||||
dispatch(
|
|
||||||
rgLayerLineAdded({
|
|
||||||
layerId: selectedLayerId,
|
|
||||||
points: [pos.x, pos.y, pos.x, pos.y],
|
|
||||||
tool,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
$isDrawing.set(true);
|
|
||||||
$lastMouseDownPos.set(pos);
|
|
||||||
} else if (tool === 'rect') {
|
|
||||||
$lastMouseDownPos.set(snapPosToStage(pos, stage));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[dispatch, selectedLayerId, selectedLayerType, tool]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onMouseUp = useCallback(
|
|
||||||
(e: KonvaEventObject<MouseEvent>) => {
|
|
||||||
const stage = e.target.getStage();
|
|
||||||
if (!stage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pos = $lastCursorPos.get();
|
|
||||||
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const lastPos = $lastMouseDownPos.get();
|
|
||||||
const tool = $tool.get();
|
|
||||||
if (lastPos && selectedLayerId && tool === 'rect') {
|
|
||||||
const snappedPos = snapPosToStage(pos, stage);
|
|
||||||
dispatch(
|
|
||||||
rgLayerRectAdded({
|
|
||||||
layerId: selectedLayerId,
|
|
||||||
rect: {
|
|
||||||
x: Math.min(snappedPos.x, lastPos.x),
|
|
||||||
y: Math.min(snappedPos.y, lastPos.y),
|
|
||||||
width: Math.abs(snappedPos.x - lastPos.x),
|
|
||||||
height: Math.abs(snappedPos.y - lastPos.y),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$isDrawing.set(false);
|
|
||||||
$lastMouseDownPos.set(null);
|
|
||||||
},
|
|
||||||
[dispatch, selectedLayerId, selectedLayerType]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onMouseMove = useCallback(
|
|
||||||
(e: KonvaEventObject<MouseEvent>) => {
|
|
||||||
const stage = e.target.getStage();
|
|
||||||
if (!stage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pos = syncCursorPos(stage);
|
|
||||||
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (getIsFocused(stage) && getIsMouseDown(e) && (tool === 'brush' || tool === 'eraser')) {
|
|
||||||
if ($isDrawing.get()) {
|
|
||||||
// Continue the last line
|
|
||||||
if (lastCursorPosRef.current) {
|
|
||||||
// Dispatching redux events impacts perf substantially - using brush spacing keeps dispatches to a reasonable number
|
|
||||||
if (Math.hypot(lastCursorPosRef.current[0] - pos.x, lastCursorPosRef.current[1] - pos.y) < brushSpacingPx) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lastCursorPosRef.current = [pos.x, pos.y];
|
|
||||||
dispatch(rgLayerPointsAdded({ layerId: selectedLayerId, point: lastCursorPosRef.current }));
|
|
||||||
} else {
|
|
||||||
// Start a new line
|
|
||||||
dispatch(rgLayerLineAdded({ layerId: selectedLayerId, points: [pos.x, pos.y, pos.x, pos.y], tool }));
|
|
||||||
}
|
|
||||||
$isDrawing.set(true);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[brushSpacingPx, dispatch, selectedLayerId, selectedLayerType, tool]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onMouseLeave = useCallback(
|
|
||||||
(e: KonvaEventObject<MouseEvent>) => {
|
|
||||||
const stage = e.target.getStage();
|
|
||||||
if (!stage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pos = syncCursorPos(stage);
|
|
||||||
$isDrawing.set(false);
|
|
||||||
$lastCursorPos.set(null);
|
|
||||||
$lastMouseDownPos.set(null);
|
|
||||||
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (getIsFocused(stage) && getIsMouseDown(e) && (tool === 'brush' || tool === 'eraser')) {
|
|
||||||
dispatch(rgLayerPointsAdded({ layerId: selectedLayerId, point: [pos.x, pos.y] }));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[selectedLayerId, selectedLayerType, tool, dispatch]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onMouseWheel = useCallback(
|
|
||||||
(e: KonvaEventObject<WheelEvent>) => {
|
|
||||||
e.evt.preventDefault();
|
|
||||||
|
|
||||||
if (selectedLayerType !== 'regional_guidance_layer' || (tool !== 'brush' && tool !== 'eraser')) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// checking for ctrl key is pressed or not,
|
|
||||||
// so that brush size can be controlled using ctrl + scroll up/down
|
|
||||||
|
|
||||||
// Invert the delta if the property is set to true
|
|
||||||
let delta = e.evt.deltaY;
|
|
||||||
if (shouldInvertBrushSizeScrollDirection) {
|
|
||||||
delta = -delta;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($ctrl.get() || $meta.get()) {
|
|
||||||
dispatch(brushSizeChanged(calculateNewBrushSize(brushSize, delta)));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[selectedLayerType, tool, shouldInvertBrushSizeScrollDirection, dispatch, brushSize]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handlers = useMemo(
|
|
||||||
() => ({ onMouseDown, onMouseUp, onMouseMove, onMouseLeave, onMouseWheel }),
|
|
||||||
[onMouseDown, onMouseUp, onMouseMove, onMouseLeave, onMouseWheel]
|
|
||||||
);
|
|
||||||
|
|
||||||
return handlers;
|
|
||||||
};
|
|
@ -36,6 +36,9 @@ import { assert } from 'tsafe';
|
|||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
|
AddLineArg,
|
||||||
|
AddPointToLineArg,
|
||||||
|
AddRectArg,
|
||||||
ControlAdapterLayer,
|
ControlAdapterLayer,
|
||||||
ControlLayersState,
|
ControlLayersState,
|
||||||
DrawingTool,
|
DrawingTool,
|
||||||
@ -492,11 +495,11 @@ export const controlLayersSlice = createSlice({
|
|||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
layer.uploadedMaskImage = null;
|
layer.uploadedMaskImage = null;
|
||||||
},
|
},
|
||||||
prepare: (payload: { layerId: string; points: [number, number, number, number]; tool: DrawingTool }) => ({
|
prepare: (payload: AddLineArg) => ({
|
||||||
payload: { ...payload, lineUuid: uuidv4() },
|
payload: { ...payload, lineUuid: uuidv4() },
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
rgLayerPointsAdded: (state, action: PayloadAction<{ layerId: string; point: [number, number] }>) => {
|
rgLayerPointsAdded: (state, action: PayloadAction<AddPointToLineArg>) => {
|
||||||
const { layerId, point } = action.payload;
|
const { layerId, point } = action.payload;
|
||||||
const layer = selectRGLayerOrThrow(state, layerId);
|
const layer = selectRGLayerOrThrow(state, layerId);
|
||||||
const lastLine = layer.maskObjects.findLast(isLine);
|
const lastLine = layer.maskObjects.findLast(isLine);
|
||||||
@ -529,7 +532,7 @@ export const controlLayersSlice = createSlice({
|
|||||||
layer.bboxNeedsUpdate = true;
|
layer.bboxNeedsUpdate = true;
|
||||||
layer.uploadedMaskImage = null;
|
layer.uploadedMaskImage = null;
|
||||||
},
|
},
|
||||||
prepare: (payload: { layerId: string; rect: IRect }) => ({ payload: { ...payload, rectUuid: uuidv4() } }),
|
prepare: (payload: AddRectArg) => ({ payload: { ...payload, rectUuid: uuidv4() } }),
|
||||||
},
|
},
|
||||||
rgLayerMaskImageUploaded: (state, action: PayloadAction<{ layerId: string; imageDTO: ImageDTO }>) => {
|
rgLayerMaskImageUploaded: (state, action: PayloadAction<{ layerId: string; imageDTO: ImageDTO }>) => {
|
||||||
const { layerId, imageDTO } = action.payload;
|
const { layerId, imageDTO } = action.payload;
|
||||||
@ -888,6 +891,16 @@ export const $lastMouseDownPos = atom<Vector2d | null>(null);
|
|||||||
export const $tool = atom<Tool>('brush');
|
export const $tool = atom<Tool>('brush');
|
||||||
export const $lastCursorPos = atom<Vector2d | null>(null);
|
export const $lastCursorPos = atom<Vector2d | null>(null);
|
||||||
export const $isPreviewVisible = atom(true);
|
export const $isPreviewVisible = atom(true);
|
||||||
|
export const $lastAddedPoint = atom<Vector2d | null>(null);
|
||||||
|
export const $brushSize = atom<number>(0);
|
||||||
|
export const $brushSpacingPx = atom<number>(0);
|
||||||
|
export const $selectedLayerId = atom<string | null>(null);
|
||||||
|
export const $selectedLayerType = atom<Layer['type'] | null>(null);
|
||||||
|
export const $shouldInvertBrushSizeScrollDirection = atom(false);
|
||||||
|
|
||||||
|
export const BRUSH_SPACING_PCT = 10;
|
||||||
|
export const MIN_BRUSH_SPACING_PX = 5;
|
||||||
|
export const MAX_BRUSH_SPACING_PX = 15;
|
||||||
|
|
||||||
// IDs for singleton Konva layers and objects
|
// IDs for singleton Konva layers and objects
|
||||||
export const TOOL_PREVIEW_LAYER_ID = 'tool_preview_layer';
|
export const TOOL_PREVIEW_LAYER_ID = 'tool_preview_layer';
|
||||||
|
@ -17,6 +17,7 @@ import {
|
|||||||
zParameterPositivePrompt,
|
zParameterPositivePrompt,
|
||||||
zParameterStrength,
|
zParameterStrength,
|
||||||
} from 'features/parameters/types/parameterSchemas';
|
} from 'features/parameters/types/parameterSchemas';
|
||||||
|
import type { IRect } from 'konva/lib/types';
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
const zTool = z.enum(['brush', 'eraser', 'move', 'rect']);
|
const zTool = z.enum(['brush', 'eraser', 'move', 'rect']);
|
||||||
@ -129,3 +130,7 @@ export type ControlLayersState = {
|
|||||||
aspectRatio: AspectRatioState;
|
aspectRatio: AspectRatioState;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type AddLineArg = { layerId: string; points: [number, number, number, number]; tool: DrawingTool };
|
||||||
|
export type AddPointToLineArg = { layerId: string; point: [number, number] };
|
||||||
|
export type AddRectArg = { layerId: string; rect: IRect };
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
|
import { calculateNewBrushSize } from 'features/canvas/hooks/useCanvasZoom';
|
||||||
import { rgbaColorToString, rgbColorToString } from 'features/canvas/util/colorToString';
|
import { rgbaColorToString, rgbColorToString } from 'features/canvas/util/colorToString';
|
||||||
import { getScaledFlooredCursorPosition, snapPosToStage } from 'features/controlLayers/hooks/mouseEventHooks';
|
|
||||||
import {
|
import {
|
||||||
$tool,
|
|
||||||
BACKGROUND_LAYER_ID,
|
BACKGROUND_LAYER_ID,
|
||||||
BACKGROUND_RECT_ID,
|
BACKGROUND_RECT_ID,
|
||||||
CA_LAYER_IMAGE_NAME,
|
CA_LAYER_IMAGE_NAME,
|
||||||
@ -31,6 +30,9 @@ import {
|
|||||||
TOOL_PREVIEW_RECT_ID,
|
TOOL_PREVIEW_RECT_ID,
|
||||||
} from 'features/controlLayers/store/controlLayersSlice';
|
} from 'features/controlLayers/store/controlLayersSlice';
|
||||||
import type {
|
import type {
|
||||||
|
AddLineArg,
|
||||||
|
AddPointToLineArg,
|
||||||
|
AddRectArg,
|
||||||
ControlAdapterLayer,
|
ControlAdapterLayer,
|
||||||
InitialImageLayer,
|
InitialImageLayer,
|
||||||
Layer,
|
Layer,
|
||||||
@ -42,8 +44,10 @@ import type {
|
|||||||
import { getLayerBboxFast, getLayerBboxPixels } from 'features/controlLayers/util/bbox';
|
import { getLayerBboxFast, getLayerBboxPixels } from 'features/controlLayers/util/bbox';
|
||||||
import { t } from 'i18next';
|
import { t } from 'i18next';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
|
import type { KonvaEventObject } from 'konva/lib/Node';
|
||||||
import type { IRect, Vector2d } from 'konva/lib/types';
|
import type { IRect, Vector2d } from 'konva/lib/types';
|
||||||
import { debounce } from 'lodash-es';
|
import { debounce } from 'lodash-es';
|
||||||
|
import type { WritableAtom } from 'nanostores';
|
||||||
import type { RgbColor } from 'react-colorful';
|
import type { RgbColor } from 'react-colorful';
|
||||||
import type { ImageDTO } from 'services/api/types';
|
import type { ImageDTO } from 'services/api/types';
|
||||||
import { assert } from 'tsafe';
|
import { assert } from 'tsafe';
|
||||||
@ -71,6 +75,46 @@ const selectVectorMaskObjects = (node: Konva.Node): boolean => {
|
|||||||
return node.name() === RG_LAYER_LINE_NAME || node.name() === RG_LAYER_RECT_NAME;
|
return node.name() === RG_LAYER_LINE_NAME || node.name() === RG_LAYER_RECT_NAME;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getIsFocused = (stage: Konva.Stage) => {
|
||||||
|
return stage.container().contains(document.activeElement);
|
||||||
|
};
|
||||||
|
const getIsMouseDown = (e: KonvaEventObject<MouseEvent>) => e.evt.buttons === 1;
|
||||||
|
|
||||||
|
const SNAP_PX = 10;
|
||||||
|
|
||||||
|
const snapPosToStage = (pos: Vector2d, stage: Konva.Stage) => {
|
||||||
|
const snappedPos = { ...pos };
|
||||||
|
// Get the normalized threshold for snapping to the edge of the stage
|
||||||
|
const thresholdX = SNAP_PX / stage.scaleX();
|
||||||
|
const thresholdY = SNAP_PX / stage.scaleY();
|
||||||
|
const stageWidth = stage.width() / stage.scaleX();
|
||||||
|
const stageHeight = stage.height() / stage.scaleY();
|
||||||
|
// Snap to the edge of the stage if within threshold
|
||||||
|
if (pos.x - thresholdX < 0) {
|
||||||
|
snappedPos.x = 0;
|
||||||
|
} else if (pos.x + thresholdX > stageWidth) {
|
||||||
|
snappedPos.x = Math.floor(stageWidth);
|
||||||
|
}
|
||||||
|
if (pos.y - thresholdY < 0) {
|
||||||
|
snappedPos.y = 0;
|
||||||
|
} else if (pos.y + thresholdY > stageHeight) {
|
||||||
|
snappedPos.y = Math.floor(stageHeight);
|
||||||
|
}
|
||||||
|
return snappedPos;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getScaledFlooredCursorPosition = (stage: Konva.Stage) => {
|
||||||
|
const pointerPosition = stage.getPointerPosition();
|
||||||
|
const stageTransform = stage.getAbsoluteTransform().copy();
|
||||||
|
if (!pointerPosition) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const scaledCursorPosition = stageTransform.invert().point(pointerPosition);
|
||||||
|
return {
|
||||||
|
x: Math.floor(scaledCursorPosition.x),
|
||||||
|
y: Math.floor(scaledCursorPosition.y),
|
||||||
|
};
|
||||||
|
};
|
||||||
/**
|
/**
|
||||||
* Creates the singleton tool preview layer and all its objects.
|
* Creates the singleton tool preview layer and all its objects.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
@ -80,25 +124,6 @@ const createToolPreviewLayer = (stage: Konva.Stage): Konva.Layer => {
|
|||||||
const toolPreviewLayer = new Konva.Layer({ id: TOOL_PREVIEW_LAYER_ID, visible: false, listening: false });
|
const toolPreviewLayer = new Konva.Layer({ id: TOOL_PREVIEW_LAYER_ID, visible: false, listening: false });
|
||||||
stage.add(toolPreviewLayer);
|
stage.add(toolPreviewLayer);
|
||||||
|
|
||||||
// Add handlers to show/hide the tool preview layer as the mouse enters/leaves the stage
|
|
||||||
stage.on('mousemove', (e) => {
|
|
||||||
const tool = $tool.get();
|
|
||||||
e.target
|
|
||||||
.getStage()
|
|
||||||
?.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)
|
|
||||||
?.visible(tool === 'brush' || tool === 'eraser');
|
|
||||||
});
|
|
||||||
stage.on('mouseleave', (e) => {
|
|
||||||
e.target.getStage()?.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)?.visible(false);
|
|
||||||
});
|
|
||||||
stage.on('mouseenter', (e) => {
|
|
||||||
const tool = $tool.get();
|
|
||||||
e.target
|
|
||||||
.getStage()
|
|
||||||
?.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)
|
|
||||||
?.visible(tool === 'brush' || tool === 'eraser');
|
|
||||||
});
|
|
||||||
|
|
||||||
// Create the brush preview group & circles
|
// Create the brush preview group & circles
|
||||||
const brushPreviewGroup = new Konva.Group({ id: TOOL_PREVIEW_BRUSH_GROUP_ID });
|
const brushPreviewGroup = new Konva.Group({ id: TOOL_PREVIEW_BRUSH_GROUP_ID });
|
||||||
const brushPreviewFill = new Konva.Circle({
|
const brushPreviewFill = new Konva.Circle({
|
||||||
@ -974,7 +999,7 @@ const createNoLayersMessageLayer = (stage: Konva.Stage): Konva.Layer => {
|
|||||||
y: 0,
|
y: 0,
|
||||||
align: 'center',
|
align: 'center',
|
||||||
verticalAlign: 'middle',
|
verticalAlign: 'middle',
|
||||||
text: t('controlLayers.noLayersAdded'),
|
text: t('controlLayers.noLayersAdded', 'No Layers Added'),
|
||||||
fontFamily: '"Inter Variable", sans-serif',
|
fontFamily: '"Inter Variable", sans-serif',
|
||||||
fontStyle: '600',
|
fontStyle: '600',
|
||||||
fill: 'white',
|
fill: 'white',
|
||||||
@ -1005,6 +1030,194 @@ const renderNoLayersMessage = (stage: Konva.Stage, layerCount: number, width: nu
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type SetStageEventHandlersArg = {
|
||||||
|
stage: Konva.Stage;
|
||||||
|
$tool: WritableAtom<Tool>;
|
||||||
|
$isDrawing: WritableAtom<boolean>;
|
||||||
|
$lastMouseDownPos: WritableAtom<Vector2d | null>;
|
||||||
|
$lastCursorPos: WritableAtom<Vector2d | null>;
|
||||||
|
$lastAddedPoint: WritableAtom<Vector2d | null>;
|
||||||
|
$brushSize: WritableAtom<number>;
|
||||||
|
$brushSpacingPx: WritableAtom<number>;
|
||||||
|
$selectedLayerId: WritableAtom<string | null>;
|
||||||
|
$selectedLayerType: WritableAtom<Layer['type'] | null>;
|
||||||
|
$shouldInvertBrushSizeScrollDirection: WritableAtom<boolean>;
|
||||||
|
onRGLayerLineAdded: (arg: AddLineArg) => void;
|
||||||
|
onRGLayerPointAddedToLine: (arg: AddPointToLineArg) => void;
|
||||||
|
onRGLayerRectAdded: (arg: AddRectArg) => void;
|
||||||
|
onBrushSizeChanged: (size: number) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setStageEventHandlers = ({
|
||||||
|
stage,
|
||||||
|
$tool,
|
||||||
|
$isDrawing,
|
||||||
|
$lastMouseDownPos,
|
||||||
|
$lastCursorPos,
|
||||||
|
$lastAddedPoint,
|
||||||
|
$brushSize,
|
||||||
|
$brushSpacingPx,
|
||||||
|
$selectedLayerId,
|
||||||
|
$selectedLayerType,
|
||||||
|
$shouldInvertBrushSizeScrollDirection,
|
||||||
|
onRGLayerLineAdded,
|
||||||
|
onRGLayerPointAddedToLine,
|
||||||
|
onRGLayerRectAdded,
|
||||||
|
onBrushSizeChanged,
|
||||||
|
}: SetStageEventHandlersArg): (() => void) => {
|
||||||
|
const syncCursorPos = (stage: Konva.Stage) => {
|
||||||
|
const pos = getScaledFlooredCursorPosition(stage);
|
||||||
|
if (!pos) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
$lastCursorPos.set(pos);
|
||||||
|
return pos;
|
||||||
|
};
|
||||||
|
|
||||||
|
stage.on('mouseenter', (e) => {
|
||||||
|
const stage = e.target.getStage();
|
||||||
|
if (!stage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const tool = $tool.get();
|
||||||
|
stage.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)?.visible(tool === 'brush' || tool === 'eraser');
|
||||||
|
});
|
||||||
|
|
||||||
|
stage.on('mousedown', (e) => {
|
||||||
|
const stage = e.target.getStage();
|
||||||
|
if (!stage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const tool = $tool.get();
|
||||||
|
const pos = syncCursorPos(stage);
|
||||||
|
const selectedLayerId = $selectedLayerId.get();
|
||||||
|
const selectedLayerType = $selectedLayerType.get();
|
||||||
|
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (tool === 'brush' || tool === 'eraser') {
|
||||||
|
onRGLayerLineAdded({
|
||||||
|
layerId: selectedLayerId,
|
||||||
|
points: [pos.x, pos.y, pos.x, pos.y],
|
||||||
|
tool,
|
||||||
|
});
|
||||||
|
$isDrawing.set(true);
|
||||||
|
$lastMouseDownPos.set(pos);
|
||||||
|
} else if (tool === 'rect') {
|
||||||
|
$lastMouseDownPos.set(snapPosToStage(pos, stage));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
stage.on('mouseup', (e) => {
|
||||||
|
const stage = e.target.getStage();
|
||||||
|
if (!stage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const pos = $lastCursorPos.get();
|
||||||
|
const selectedLayerId = $selectedLayerId.get();
|
||||||
|
const selectedLayerType = $selectedLayerType.get();
|
||||||
|
|
||||||
|
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const lastPos = $lastMouseDownPos.get();
|
||||||
|
const tool = $tool.get();
|
||||||
|
if (lastPos && selectedLayerId && tool === 'rect') {
|
||||||
|
const snappedPos = snapPosToStage(pos, stage);
|
||||||
|
onRGLayerRectAdded({
|
||||||
|
layerId: selectedLayerId,
|
||||||
|
rect: {
|
||||||
|
x: Math.min(snappedPos.x, lastPos.x),
|
||||||
|
y: Math.min(snappedPos.y, lastPos.y),
|
||||||
|
width: Math.abs(snappedPos.x - lastPos.x),
|
||||||
|
height: Math.abs(snappedPos.y - lastPos.y),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
$isDrawing.set(false);
|
||||||
|
$lastMouseDownPos.set(null);
|
||||||
|
});
|
||||||
|
|
||||||
|
stage.on('mousemove', (e) => {
|
||||||
|
const stage = e.target.getStage();
|
||||||
|
if (!stage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const tool = $tool.get();
|
||||||
|
const pos = syncCursorPos(stage);
|
||||||
|
const selectedLayerId = $selectedLayerId.get();
|
||||||
|
const selectedLayerType = $selectedLayerType.get();
|
||||||
|
|
||||||
|
stage.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)?.visible(tool === 'brush' || tool === 'eraser');
|
||||||
|
|
||||||
|
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (getIsFocused(stage) && getIsMouseDown(e) && (tool === 'brush' || tool === 'eraser')) {
|
||||||
|
if ($isDrawing.get()) {
|
||||||
|
// Continue the last line
|
||||||
|
const lastAddedPoint = $lastAddedPoint.get();
|
||||||
|
if (lastAddedPoint) {
|
||||||
|
// Dispatching redux events impacts perf substantially - using brush spacing keeps dispatches to a reasonable number
|
||||||
|
if (Math.hypot(lastAddedPoint.x - pos.x, lastAddedPoint.y - pos.y) < $brushSpacingPx.get()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$lastAddedPoint.set({ x: pos.x, y: pos.y });
|
||||||
|
onRGLayerPointAddedToLine({ layerId: selectedLayerId, point: [pos.x, pos.y] });
|
||||||
|
} else {
|
||||||
|
// Start a new line
|
||||||
|
onRGLayerLineAdded({ layerId: selectedLayerId, points: [pos.x, pos.y, pos.x, pos.y], tool });
|
||||||
|
}
|
||||||
|
$isDrawing.set(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
stage.on('mouseleave', (e) => {
|
||||||
|
const stage = e.target.getStage();
|
||||||
|
if (!stage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const pos = syncCursorPos(stage);
|
||||||
|
$isDrawing.set(false);
|
||||||
|
$lastCursorPos.set(null);
|
||||||
|
$lastMouseDownPos.set(null);
|
||||||
|
const selectedLayerId = $selectedLayerId.get();
|
||||||
|
const selectedLayerType = $selectedLayerType.get();
|
||||||
|
const tool = $tool.get();
|
||||||
|
|
||||||
|
stage.findOne<Konva.Layer>(`#${TOOL_PREVIEW_LAYER_ID}`)?.visible(false);
|
||||||
|
|
||||||
|
if (!pos || !selectedLayerId || selectedLayerType !== 'regional_guidance_layer') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (getIsFocused(stage) && getIsMouseDown(e) && (tool === 'brush' || tool === 'eraser')) {
|
||||||
|
onRGLayerPointAddedToLine({ layerId: selectedLayerId, point: [pos.x, pos.y] });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
stage.on('wheel', (e) => {
|
||||||
|
e.evt.preventDefault();
|
||||||
|
const selectedLayerType = $selectedLayerType.get();
|
||||||
|
const tool = $tool.get();
|
||||||
|
if (selectedLayerType !== 'regional_guidance_layer' || (tool !== 'brush' && tool !== 'eraser')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Invert the delta if the property is set to true
|
||||||
|
let delta = e.evt.deltaY;
|
||||||
|
if ($shouldInvertBrushSizeScrollDirection.get()) {
|
||||||
|
delta = -delta;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.evt.ctrlKey || e.evt.metaKey) {
|
||||||
|
onBrushSizeChanged(calculateNewBrushSize($brushSize.get(), delta));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => stage.off('mousedown mouseup mousemove mouseenter mouseleave wheel');
|
||||||
|
};
|
||||||
|
|
||||||
export const renderers = {
|
export const renderers = {
|
||||||
renderToolPreview,
|
renderToolPreview,
|
||||||
renderLayers,
|
renderLayers,
|
||||||
|
Loading…
Reference in New Issue
Block a user