feat(ui): re-implement with imperative konva api (wip)

This commit is contained in:
psychedelicious 2024-04-17 11:57:57 +10:00 committed by Kent Keirsey
parent bbbb5479e8
commit 525e6d697c
2 changed files with 111 additions and 69 deletions

View File

@ -6,6 +6,11 @@ import { rgbColorToString } from 'features/canvas/util/colorToString';
import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition'; import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition';
import { import {
$cursorPosition, $cursorPosition,
BRUSH_PREVIEW_BORDER_INNER_ID,
BRUSH_PREVIEW_BORDER_OUTER_ID,
BRUSH_PREVIEW_FILL_ID,
BRUSH_PREVIEW_LAYER_ID,
getPromptRegionLayerObjectGroupId,
layerBboxChanged, layerBboxChanged,
layerSelected, layerSelected,
layerTranslated, layerTranslated,
@ -17,6 +22,7 @@ import Konva from 'konva';
import type { Node, NodeConfig } from 'konva/lib/Node'; import type { Node, NodeConfig } from 'konva/lib/Node';
import { atom } from 'nanostores'; import { atom } from 'nanostores';
import { useLayoutEffect } from 'react'; import { useLayoutEffect } from 'react';
import { v4 as uuidv4 } from 'uuid';
import { useMouseDown, useMouseEnter, useMouseLeave, useMouseMove, useMouseUp } from './mouseEventHooks'; import { useMouseDown, useMouseEnter, useMouseLeave, useMouseMove, useMouseUp } from './mouseEventHooks';
@ -29,10 +35,6 @@ type Props = {
export const selectPromptLayerObjectGroup = (item: Node<NodeConfig>) => export const selectPromptLayerObjectGroup = (item: Node<NodeConfig>) =>
item.name() !== REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME; item.name() !== REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME;
const BRUSH_PREVIEW_FILL = 'brushPreviewFill';
const BRUSH_PREVIEW_OUTLINE_INNER = 'brushPreviewOutlineInner';
const BRUSH_PREVIEW_OUTLINE_OUTER = 'brushPreviewOutlineOuter';
const isKonvaLayer = (node: Node<NodeConfig>): node is Konva.Layer => node.nodeType === 'Layer'; const isKonvaLayer = (node: Node<NodeConfig>): node is Konva.Layer => node.nodeType === 'Layer';
const isKonvaLine = (node: Node<NodeConfig>): node is Konva.Line => node.nodeType === 'Line'; const isKonvaLine = (node: Node<NodeConfig>): node is Konva.Line => node.nodeType === 'Line';
const isKonvaGroup = (node: Node<NodeConfig>): node is Konva.Group => node.nodeType === 'Group'; const isKonvaGroup = (node: Node<NodeConfig>): node is Konva.Group => node.nodeType === 'Group';
@ -41,8 +43,8 @@ const isKonvaRect = (node: Node<NodeConfig>): node is Konva.Rect => node.nodeTyp
const $brushPreviewNodes = atom<{ const $brushPreviewNodes = atom<{
layer: Konva.Layer; layer: Konva.Layer;
fill: Konva.Circle; fill: Konva.Circle;
outlineInner: Konva.Circle; borderInner: Konva.Circle;
outlineOuter: Konva.Circle; borderOuter: Konva.Circle;
} | null>(null); } | null>(null);
export const LogicalStage = (props: Props) => { export const LogicalStage = (props: Props) => {
@ -68,32 +70,32 @@ export const LogicalStage = (props: Props) => {
container: props.container, container: props.container,
}); });
const brushPreviewLayer = new Konva.Layer({ id: 'brushPreviewLayer' }); const brushPreviewLayer = new Konva.Layer({ id: BRUSH_PREVIEW_LAYER_ID });
stage.add(brushPreviewLayer); stage.add(brushPreviewLayer);
const fill = new Konva.Circle({ const fill = new Konva.Circle({
id: BRUSH_PREVIEW_FILL, id: BRUSH_PREVIEW_FILL_ID,
listening: false, listening: false,
strokeEnabled: false, strokeEnabled: false,
strokeHitEnabled: false, strokeHitEnabled: false,
}); });
const outlineInner = new Konva.Circle({ const borderInner = new Konva.Circle({
id: BRUSH_PREVIEW_OUTLINE_INNER, id: BRUSH_PREVIEW_BORDER_INNER_ID,
listening: false, listening: false,
stroke: 'rgba(0,0,0,1)', stroke: 'rgba(0,0,0,1)',
strokeWidth: 1, strokeWidth: 1,
strokeEnabled: true, strokeEnabled: true,
}); });
const outlineOuter = new Konva.Circle({ const borderOuter = new Konva.Circle({
id: BRUSH_PREVIEW_OUTLINE_OUTER, id: BRUSH_PREVIEW_BORDER_OUTER_ID,
listening: false, listening: false,
stroke: 'rgba(255,255,255,0.8)', stroke: 'rgba(255,255,255,0.8)',
strokeWidth: 1, strokeWidth: 1,
strokeEnabled: true, strokeEnabled: true,
}); });
brushPreviewLayer.add(fill); brushPreviewLayer.add(fill);
brushPreviewLayer.add(outlineInner); brushPreviewLayer.add(borderInner);
brushPreviewLayer.add(outlineOuter); brushPreviewLayer.add(borderOuter);
$brushPreviewNodes.set({ layer: brushPreviewLayer, fill, outlineInner, outlineOuter }); $brushPreviewNodes.set({ layer: brushPreviewLayer, fill, borderInner: borderInner, borderOuter: borderOuter });
$stage.set(stage); $stage.set(stage);
@ -156,8 +158,8 @@ export const LogicalStage = (props: Props) => {
fill, fill,
globalCompositeOperation: state.tool === 'brush' ? 'source-over' : 'destination-out', globalCompositeOperation: state.tool === 'brush' ? 'source-over' : 'destination-out',
}); });
brushPreviewNodes.outlineInner.setAttrs({ x: cursorPosition.x, y: cursorPosition.y, radius: state.brushSize / 2 }); brushPreviewNodes.borderInner.setAttrs({ x: cursorPosition.x, y: cursorPosition.y, radius: state.brushSize / 2 });
brushPreviewNodes.outlineOuter.setAttrs({ brushPreviewNodes.borderOuter.setAttrs({
x: cursorPosition.x, x: cursorPosition.x,
y: cursorPosition.y, y: cursorPosition.y,
radius: state.brushSize / 2 + 1, radius: state.brushSize / 2 + 1,
@ -189,6 +191,8 @@ export const LogicalStage = (props: Props) => {
name: REGIONAL_PROMPT_LAYER_NAME, name: REGIONAL_PROMPT_LAYER_NAME,
draggable: true, draggable: true,
listening: reduxLayer.id === state.selectedLayer, listening: reduxLayer.id === state.selectedLayer,
x: reduxLayer.x,
y: reduxLayer.y,
}); });
konvaLayer.on('dragmove', function (e) { konvaLayer.on('dragmove', function (e) {
dispatch( dispatch(
@ -212,12 +216,22 @@ export const LogicalStage = (props: Props) => {
return pos; return pos;
}); });
stage.add(konvaLayer); stage.add(konvaLayer);
konvaLayer.add(
new Konva.Group({
id: getPromptRegionLayerObjectGroupId(reduxLayer.id, uuidv4()),
name: REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME,
listening: false,
})
);
$brushPreviewNodes.get()?.layer.moveToTop(); $brushPreviewNodes.get()?.layer.moveToTop();
} else { } else {
konvaLayer.listening(reduxLayer.id === state.selectedLayer); konvaLayer.listening(reduxLayer.id === state.selectedLayer);
konvaLayer.x(reduxLayer.x);
konvaLayer.y(reduxLayer.y);
} }
const color = rgbColorToString(reduxLayer.color); const color = rgbColorToString(reduxLayer.color);
const konvaObjectGroup = konvaLayer.findOne(`.${REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME}`) as Konva.Group;
// Remove deleted objects // Remove deleted objects
const objectIds = reduxLayer.objects.map((o) => o.id); const objectIds = reduxLayer.objects.map((o) => o.id);
@ -228,35 +242,41 @@ export const LogicalStage = (props: Props) => {
} }
for (const reduxObject of reduxLayer.objects) { for (const reduxObject of reduxLayer.objects) {
// TODO: Handle rects, images, etc
if (reduxObject.kind !== 'line') { if (reduxObject.kind !== 'line') {
return; return;
} }
let konvaObject = stage.findOne(`#${reduxObject.id}`) as Konva.Line | undefined; const konvaObject = stage.findOne(`#${reduxObject.id}`) as Konva.Line | undefined;
if (!konvaObject) { if (!konvaObject) {
konvaObject = new Konva.Line({ // This object hasn't been added to the konva state yet.
id: reduxObject.id, konvaObjectGroup.add(
key: reduxObject.id, new Konva.Line({
name: `${reduxLayer.id}-object`, id: reduxObject.id,
points: reduxObject.points, key: reduxObject.id,
strokeWidth: reduxObject.strokeWidth, name: `${reduxLayer.id}-object`,
stroke: color, points: reduxObject.points,
tension: 0, strokeWidth: reduxObject.strokeWidth,
lineCap: 'round', stroke: color,
lineJoin: 'round', tension: 0,
shadowForStrokeEnabled: false, lineCap: 'round',
globalCompositeOperation: reduxObject.tool === 'brush' ? 'source-over' : 'destination-out', lineJoin: 'round',
listening: false, shadowForStrokeEnabled: false,
visible: reduxLayer.isVisible, globalCompositeOperation: reduxObject.tool === 'brush' ? 'source-over' : 'destination-out',
}); listening: false,
konvaLayer.add(konvaObject); visible: reduxLayer.isVisible,
})
);
} else { } else {
// Only update the points if they have changed. The point values are never mutated, they are only added to the array.
if (konvaObject.points().length !== reduxObject.points.length) { if (konvaObject.points().length !== reduxObject.points.length) {
konvaObject.points(reduxObject.points); konvaObject.points(reduxObject.points);
} }
// Only update the color if it has changed.
if (konvaObject.stroke() !== color) { if (konvaObject.stroke() !== color) {
konvaObject.stroke(color); konvaObject.stroke(color);
} }
// Only update layer visibility if it has changed.
if (konvaObject.visible() !== reduxLayer.isVisible) { if (konvaObject.visible() !== reduxLayer.isVisible) {
konvaObject.visible(reduxLayer.isVisible); konvaObject.visible(reduxLayer.isVisible);
} }
@ -269,6 +289,14 @@ export const LogicalStage = (props: Props) => {
if (!stage) { if (!stage) {
return; return;
} }
stage.container().style.cursor = state.tool === 'move' ? 'default' : 'none';
}, [stage, state.tool]);
useLayoutEffect(() => {
console.log('bbox effect');
if (!stage) {
return;
}
if (state.tool !== 'move') { if (state.tool !== 'move') {
// Tool was just changed to something other than move - hide all layer bounding boxes // Tool was just changed to something other than move - hide all layer bounding boxes

View File

@ -83,12 +83,22 @@ export const regionalPromptsSlice = createSlice({
initialState: initialRegionalPromptsState, initialState: initialRegionalPromptsState,
reducers: { reducers: {
layerAdded: { layerAdded: {
reducer: (state, action: PayloadAction<Layer['kind'], string, { id: string }>) => { reducer: (state, action: PayloadAction<Layer['kind'], string, { uuid: string; color: RgbColor }>) => {
const newLayer = buildLayer(action.meta.id, action.payload, state.layers.length); const layer: PromptRegionLayer = {
state.layers.push(newLayer); id: getPromptRegionLayerId(action.meta.uuid),
state.selectedLayer = newLayer.id; isVisible: true,
bbox: null,
kind: action.payload,
prompt: '',
objects: [],
color: action.meta.color,
x: 0,
y: 0,
};
state.layers.push(layer);
state.selectedLayer = layer.id;
}, },
prepare: (payload: Layer['kind']) => ({ payload, meta: { id: uuidv4() } }), prepare: (payload: Layer['kind']) => ({ payload, meta: { uuid: uuidv4(), color: LayerColors.next() } }),
}, },
layerSelected: (state, action: PayloadAction<string>) => { layerSelected: (state, action: PayloadAction<string>) => {
state.selectedLayer = action.payload; state.selectedLayer = action.payload;
@ -166,20 +176,21 @@ export const regionalPromptsSlice = createSlice({
layer.color = color; layer.color = color;
}, },
lineAdded: { lineAdded: {
reducer: (state, action: PayloadAction<[number, number], string, { id: string }>) => { reducer: (state, action: PayloadAction<[number, number], string, { uuid: string }>) => {
const layer = state.layers.find((l) => l.id === state.selectedLayer); const layer = state.layers.find((l) => l.id === state.selectedLayer);
if (!layer || layer.kind !== 'promptRegionLayer') { if (!layer || layer.kind !== 'promptRegionLayer') {
return; return;
} }
const lineId = getPromptRegionLayerLineId(layer.id, action.meta.uuid);
layer.objects.push({ layer.objects.push({
kind: 'line', kind: 'line',
tool: state.tool, tool: state.tool,
id: action.meta.id, id: lineId,
points: [action.payload[0] - layer.x, action.payload[1] - layer.y], points: [action.payload[0] - layer.x, action.payload[1] - layer.y],
strokeWidth: state.brushSize, strokeWidth: state.brushSize,
}); });
}, },
prepare: (payload: [number, number]) => ({ payload, meta: { id: uuidv4() } }), prepare: (payload: [number, number]) => ({ payload, meta: { uuid: uuidv4() } }),
}, },
pointsAdded: (state, action: PayloadAction<[number, number]>) => { pointsAdded: (state, action: PayloadAction<[number, number]>) => {
const layer = state.layers.find((l) => l.id === state.selectedLayer); const layer = state.layers.find((l) => l.id === state.selectedLayer);
@ -204,33 +215,29 @@ export const regionalPromptsSlice = createSlice({
}, },
}); });
const DEFAULT_COLORS = [ /**
{ r: 200, g: 0, b: 0 }, * This class is used to cycle through a set of colors for the prompt region layers.
{ r: 0, g: 200, b: 0 }, */
{ r: 0, g: 0, b: 200 }, class LayerColors {
{ r: 200, g: 200, b: 0 }, static COLORS: RgbColor[] = [
{ r: 0, g: 200, b: 200 }, { r: 200, g: 0, b: 0 },
{ r: 200, g: 0, b: 200 }, { r: 0, g: 200, b: 0 },
]; { r: 0, g: 0, b: 200 },
{ r: 200, g: 200, b: 0 },
const buildLayer = (id: string, kind: Layer['kind'], layerCount: number): Layer => { { r: 0, g: 200, b: 200 },
if (kind === 'promptRegionLayer') { { r: 200, g: 0, b: 200 },
const color = DEFAULT_COLORS[layerCount % DEFAULT_COLORS.length]; ];
assert(color, 'Color not found'); static i = this.COLORS.length - 1;
return { /**
id, * Get the next color in the sequence.
isVisible: true, */
bbox: null, static next(): RgbColor {
kind, this.i = (this.i + 1) % this.COLORS.length;
prompt: '', const color = this.COLORS[this.i];
objects: [], assert(color);
color, return color;
x: 0,
y: 0,
};
} }
assert(false, `Unknown layer kind: ${kind}`); }
};
export const { export const {
layerAdded, layerAdded,
@ -276,5 +283,12 @@ export const getStage = (): Stage => {
assert(stage); assert(stage);
return stage; return stage;
}; };
export const BRUSH_PREVIEW_LAYER_ID = 'brushPreviewLayer';
export const BRUSH_PREVIEW_FILL_ID = 'brushPreviewFill';
export const BRUSH_PREVIEW_BORDER_INNER_ID = 'brushPreviewBorderInner';
export const BRUSH_PREVIEW_BORDER_OUTER_ID = 'brushPreviewBorderOuter';
export const REGIONAL_PROMPT_LAYER_NAME = 'regionalPromptLayer'; export const REGIONAL_PROMPT_LAYER_NAME = 'regionalPromptLayer';
export const REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME = 'regionalPromptLayerObjectGroup'; export const REGIONAL_PROMPT_LAYER_OBJECT_GROUP_NAME = 'regionalPromptLayerObjectGroup';
export const getPromptRegionLayerId = (layerId: string) => `layer_${layerId}`;
export const getPromptRegionLayerLineId = (layerId: string, lineId: string) => `${layerId}.line_${lineId}`;
export const getPromptRegionLayerObjectGroupId = (layerId: string, groupId: string) => `${layerId}.objectGroup_${groupId}`;