mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
refactor(ui): create entity to konva node map abstraction (wip)
Instead of chaining konva `find` and `findOne` methods, all konva nodes are added to a mapping object. Finding and manipulating them is much simpler. Done for regions and layers, wip for control adapters.
This commit is contained in:
parent
5ff5af3ba2
commit
dd09723a2a
@ -0,0 +1,104 @@
|
|||||||
|
import type { BrushLine, EraserLine, ImageObject, RectShape } from 'features/controlLayers/store/types';
|
||||||
|
import type Konva from 'konva';
|
||||||
|
|
||||||
|
export type BrushLineEntry = {
|
||||||
|
id: string;
|
||||||
|
type: BrushLine['type'];
|
||||||
|
konvaLine: Konva.Line;
|
||||||
|
konvaLineGroup: Konva.Group;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type EraserLineEntry = {
|
||||||
|
id: string;
|
||||||
|
type: EraserLine['type'];
|
||||||
|
konvaLine: Konva.Line;
|
||||||
|
konvaLineGroup: Konva.Group;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type RectShapeEntry = {
|
||||||
|
id: string;
|
||||||
|
type: RectShape['type'];
|
||||||
|
konvaRect: Konva.Rect;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ImageEntry = {
|
||||||
|
id: string;
|
||||||
|
type: ImageObject['type'];
|
||||||
|
konvaImage: Konva.Image | null; // The image is loaded asynchronously, so it may not be available immediately
|
||||||
|
konvaGroup: Konva.Group;
|
||||||
|
};
|
||||||
|
|
||||||
|
type Entry = BrushLineEntry | EraserLineEntry | RectShapeEntry | ImageEntry;
|
||||||
|
|
||||||
|
export class EntityToKonvaMap {
|
||||||
|
mappings: Record<string, EntityToKonvaMapping>;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.mappings = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
addMapping(id: string, konvaLayer: Konva.Layer, konvaObjectGroup: Konva.Group): EntityToKonvaMapping {
|
||||||
|
const mapping = new EntityToKonvaMapping(id, konvaLayer, konvaObjectGroup);
|
||||||
|
this.mappings[id] = mapping;
|
||||||
|
return mapping;
|
||||||
|
}
|
||||||
|
|
||||||
|
getMapping(id: string): EntityToKonvaMapping | undefined {
|
||||||
|
return this.mappings[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
getMappings(): EntityToKonvaMapping[] {
|
||||||
|
return Object.values(this.mappings);
|
||||||
|
}
|
||||||
|
|
||||||
|
destroyMapping(id: string): void {
|
||||||
|
const mapping = this.getMapping(id);
|
||||||
|
if (!mapping) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
mapping.konvaObjectGroup.destroy();
|
||||||
|
delete this.mappings[id];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class EntityToKonvaMapping {
|
||||||
|
id: string;
|
||||||
|
konvaLayer: Konva.Layer;
|
||||||
|
konvaObjectGroup: Konva.Group;
|
||||||
|
konvaNodeEntries: Record<string, Entry>;
|
||||||
|
|
||||||
|
constructor(id: string, konvaLayer: Konva.Layer, konvaObjectGroup: Konva.Group) {
|
||||||
|
this.id = id;
|
||||||
|
this.konvaLayer = konvaLayer;
|
||||||
|
this.konvaObjectGroup = konvaObjectGroup;
|
||||||
|
this.konvaNodeEntries = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
addEntry<T extends Entry>(entry: T): T {
|
||||||
|
this.konvaNodeEntries[entry.id] = entry;
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
getEntry<T extends Entry>(id: string): T | undefined {
|
||||||
|
return this.konvaNodeEntries[id] as T | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
getEntries(): Entry[] {
|
||||||
|
return Object.values(this.konvaNodeEntries);
|
||||||
|
}
|
||||||
|
|
||||||
|
destroyEntry(id: string): void {
|
||||||
|
const entry = this.getEntry(id);
|
||||||
|
if (!entry) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (entry.type === 'brush_line' || entry.type === 'eraser_line') {
|
||||||
|
entry.konvaLineGroup.destroy();
|
||||||
|
} else if (entry.type === 'rect_shape') {
|
||||||
|
entry.konvaRect.destroy();
|
||||||
|
} else if (entry.type === 'image') {
|
||||||
|
entry.konvaGroup.destroy();
|
||||||
|
}
|
||||||
|
delete this.konvaNodeEntries[id];
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,5 @@
|
|||||||
import { LightnessToAlphaFilter } from 'features/controlLayers/konva/filters';
|
import { LightnessToAlphaFilter } from 'features/controlLayers/konva/filters';
|
||||||
|
import type { EntityToKonvaMap } from 'features/controlLayers/konva/konvaMap';
|
||||||
import { CA_LAYER_IMAGE_NAME, CA_LAYER_NAME, getCAImageId } from 'features/controlLayers/konva/naming';
|
import { CA_LAYER_IMAGE_NAME, CA_LAYER_NAME, getCAImageId } from 'features/controlLayers/konva/naming';
|
||||||
import type { ControlAdapterEntity } from 'features/controlLayers/store/types';
|
import type { ControlAdapterEntity } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
@ -34,6 +35,7 @@ const createCALayerImage = (konvaLayer: Konva.Layer, imageEl: HTMLImageElement):
|
|||||||
const konvaImage = new Konva.Image({
|
const konvaImage = new Konva.Image({
|
||||||
name: CA_LAYER_IMAGE_NAME,
|
name: CA_LAYER_IMAGE_NAME,
|
||||||
image: imageEl,
|
image: imageEl,
|
||||||
|
listening: false,
|
||||||
});
|
});
|
||||||
konvaLayer.add(konvaImage);
|
konvaLayer.add(konvaImage);
|
||||||
return konvaImage;
|
return konvaImage;
|
||||||
@ -128,6 +130,7 @@ const updateCALayerImageAttrs = (stage: Konva.Stage, konvaImage: Konva.Image, ca
|
|||||||
*/
|
*/
|
||||||
export const renderCALayer = (
|
export const renderCALayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
controlAdapterMap: EntityToKonvaMap,
|
||||||
ca: ControlAdapterEntity,
|
ca: ControlAdapterEntity,
|
||||||
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
||||||
): void => {
|
): void => {
|
||||||
@ -157,16 +160,17 @@ export const renderCALayer = (
|
|||||||
|
|
||||||
export const renderControlAdapters = (
|
export const renderControlAdapters = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
controlAdapterMap: EntityToKonvaMap,
|
||||||
controlAdapters: ControlAdapterEntity[],
|
controlAdapters: ControlAdapterEntity[],
|
||||||
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
getImageDTO: (imageName: string) => Promise<ImageDTO | null>
|
||||||
): void => {
|
): void => {
|
||||||
// Destroy nonexistent layers
|
// Destroy nonexistent layers
|
||||||
for (const konvaLayer of stage.find<Konva.Layer>(`.${CA_LAYER_NAME}`)) {
|
for (const mapping of controlAdapterMap.getMappings()) {
|
||||||
if (!controlAdapters.find((ca) => ca.id === konvaLayer.id())) {
|
if (!controlAdapters.find((ca) => ca.id === mapping.id)) {
|
||||||
konvaLayer.destroy();
|
controlAdapterMap.destroyMapping(mapping.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const ca of controlAdapters) {
|
for (const ca of controlAdapters) {
|
||||||
renderCALayer(stage, ca, getImageDTO);
|
renderCALayer(stage, controlAdapterMap, ca, getImageDTO);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -1,4 +1,11 @@
|
|||||||
import { rgbaColorToString } from 'common/util/colorCodeTransformers';
|
import { rgbaColorToString } from 'common/util/colorCodeTransformers';
|
||||||
|
import type {
|
||||||
|
BrushLineEntry,
|
||||||
|
EntityToKonvaMapping,
|
||||||
|
EraserLineEntry,
|
||||||
|
ImageEntry,
|
||||||
|
RectShapeEntry,
|
||||||
|
} from 'features/controlLayers/konva/konvaMap';
|
||||||
import {
|
import {
|
||||||
getLayerBboxId,
|
getLayerBboxId,
|
||||||
getObjectGroupId,
|
getObjectGroupId,
|
||||||
@ -23,18 +30,17 @@ import { v4 as uuidv4 } from 'uuid';
|
|||||||
* @param layerObjectGroup The konva layer's object group to add the line to
|
* @param layerObjectGroup The konva layer's object group to add the line to
|
||||||
* @param name The konva name for the line
|
* @param name The konva name for the line
|
||||||
*/
|
*/
|
||||||
export const getBrushLine = (brushLine: BrushLine, layerObjectGroup: Konva.Group, name: string): Konva.Line => {
|
export const getBrushLine = (mapping: EntityToKonvaMapping, brushLine: BrushLine, name: string): BrushLineEntry => {
|
||||||
let konvaLineGroup = layerObjectGroup.findOne<Konva.Group>(`#${brushLine.id}_group`);
|
let entry = mapping.getEntry<BrushLineEntry>(brushLine.id);
|
||||||
let konvaLine = konvaLineGroup?.findOne<Konva.Line>(`#${brushLine.id}`);
|
if (entry) {
|
||||||
if (konvaLine) {
|
return entry;
|
||||||
return konvaLine;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
konvaLineGroup = new Konva.Group({
|
const konvaLineGroup = new Konva.Group({
|
||||||
id: `${brushLine.id}_group`,
|
clip: brushLine.clip,
|
||||||
// clip: brushLine.clip,
|
listening: false,
|
||||||
});
|
});
|
||||||
konvaLine = new Konva.Line({
|
const konvaLine = new Konva.Line({
|
||||||
id: brushLine.id,
|
id: brushLine.id,
|
||||||
name,
|
name,
|
||||||
strokeWidth: brushLine.strokeWidth,
|
strokeWidth: brushLine.strokeWidth,
|
||||||
@ -47,8 +53,9 @@ export const getBrushLine = (brushLine: BrushLine, layerObjectGroup: Konva.Group
|
|||||||
stroke: rgbaColorToString(brushLine.color),
|
stroke: rgbaColorToString(brushLine.color),
|
||||||
});
|
});
|
||||||
konvaLineGroup.add(konvaLine);
|
konvaLineGroup.add(konvaLine);
|
||||||
layerObjectGroup.add(konvaLineGroup);
|
mapping.konvaObjectGroup.add(konvaLineGroup);
|
||||||
return konvaLine;
|
entry = mapping.addEntry({ id: brushLine.id, type: 'brush_line', konvaLine, konvaLineGroup });
|
||||||
|
return entry;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -57,10 +64,18 @@ export const getBrushLine = (brushLine: BrushLine, layerObjectGroup: Konva.Group
|
|||||||
* @param layerObjectGroup The konva layer's object group to add the line to
|
* @param layerObjectGroup The konva layer's object group to add the line to
|
||||||
* @param name The konva name for the line
|
* @param name The konva name for the line
|
||||||
*/
|
*/
|
||||||
export const getEraserLine = (eraserLine: EraserLine, layerObjectGroup: Konva.Group, name: string): Konva.Line => {
|
export const getEraserLine = (mapping: EntityToKonvaMapping, eraserLine: EraserLine, name: string): EraserLineEntry => {
|
||||||
|
let entry = mapping.getEntry<EraserLineEntry>(eraserLine.id);
|
||||||
|
if (entry) {
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
const konvaLineGroup = new Konva.Group({
|
||||||
|
clip: eraserLine.clip,
|
||||||
|
listening: false,
|
||||||
|
});
|
||||||
const konvaLine = new Konva.Line({
|
const konvaLine = new Konva.Line({
|
||||||
id: eraserLine.id,
|
id: eraserLine.id,
|
||||||
key: eraserLine.id,
|
|
||||||
name,
|
name,
|
||||||
strokeWidth: eraserLine.strokeWidth,
|
strokeWidth: eraserLine.strokeWidth,
|
||||||
tension: 0,
|
tension: 0,
|
||||||
@ -72,8 +87,10 @@ export const getEraserLine = (eraserLine: EraserLine, layerObjectGroup: Konva.Gr
|
|||||||
stroke: rgbaColorToString(DEFAULT_RGBA_COLOR),
|
stroke: rgbaColorToString(DEFAULT_RGBA_COLOR),
|
||||||
clip: eraserLine.clip,
|
clip: eraserLine.clip,
|
||||||
});
|
});
|
||||||
layerObjectGroup.add(konvaLine);
|
konvaLineGroup.add(konvaLine);
|
||||||
return konvaLine;
|
mapping.konvaObjectGroup.add(konvaLineGroup);
|
||||||
|
entry = mapping.addEntry({ id: eraserLine.id, type: 'eraser_line', konvaLine, konvaLineGroup });
|
||||||
|
return entry;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -82,7 +99,11 @@ export const getEraserLine = (eraserLine: EraserLine, layerObjectGroup: Konva.Gr
|
|||||||
* @param layerObjectGroup The konva layer's object group to add the rect to
|
* @param layerObjectGroup The konva layer's object group to add the rect to
|
||||||
* @param name The konva name for the rect
|
* @param name The konva name for the rect
|
||||||
*/
|
*/
|
||||||
export const createRectShape = (rectShape: RectShape, layerObjectGroup: Konva.Group, name: string): Konva.Rect => {
|
export const getRectShape = (mapping: EntityToKonvaMapping, rectShape: RectShape, name: string): RectShapeEntry => {
|
||||||
|
let entry = mapping.getEntry<RectShapeEntry>(rectShape.id);
|
||||||
|
if (entry) {
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
const konvaRect = new Konva.Rect({
|
const konvaRect = new Konva.Rect({
|
||||||
id: rectShape.id,
|
id: rectShape.id,
|
||||||
key: rectShape.id,
|
key: rectShape.id,
|
||||||
@ -94,8 +115,9 @@ export const createRectShape = (rectShape: RectShape, layerObjectGroup: Konva.Gr
|
|||||||
listening: false,
|
listening: false,
|
||||||
fill: rgbaColorToString(rectShape.color),
|
fill: rgbaColorToString(rectShape.color),
|
||||||
});
|
});
|
||||||
layerObjectGroup.add(konvaRect);
|
mapping.konvaObjectGroup.add(konvaRect);
|
||||||
return konvaRect;
|
entry = mapping.addEntry({ id: rectShape.id, type: 'rect_shape', konvaRect });
|
||||||
|
return entry;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -112,6 +134,7 @@ const createImagePlaceholderGroup = (
|
|||||||
fill: 'hsl(220 12% 45% / 1)', // 'base.500'
|
fill: 'hsl(220 12% 45% / 1)', // 'base.500'
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
|
listening: false,
|
||||||
});
|
});
|
||||||
const konvaPlaceholderText = new Konva.Text({
|
const konvaPlaceholderText = new Konva.Text({
|
||||||
name: 'image-placeholder-text',
|
name: 'image-placeholder-text',
|
||||||
@ -150,14 +173,20 @@ const createImagePlaceholderGroup = (
|
|||||||
* @returns A promise that resolves to the konva group for the image object
|
* @returns A promise that resolves to the konva group for the image object
|
||||||
*/
|
*/
|
||||||
export const createImageObjectGroup = async (
|
export const createImageObjectGroup = async (
|
||||||
|
mapping: EntityToKonvaMapping,
|
||||||
imageObject: ImageObject,
|
imageObject: ImageObject,
|
||||||
layerObjectGroup: Konva.Group,
|
|
||||||
name: string
|
name: string
|
||||||
): Promise<Konva.Group> => {
|
): Promise<ImageEntry> => {
|
||||||
|
let entry = mapping.getEntry<ImageEntry>(imageObject.id);
|
||||||
|
if (entry) {
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
const konvaImageGroup = new Konva.Group({ id: imageObject.id, name, listening: false });
|
const konvaImageGroup = new Konva.Group({ id: imageObject.id, name, listening: false });
|
||||||
const placeholder = createImagePlaceholderGroup(imageObject);
|
const placeholder = createImagePlaceholderGroup(imageObject);
|
||||||
konvaImageGroup.add(placeholder.konvaPlaceholderGroup);
|
konvaImageGroup.add(placeholder.konvaPlaceholderGroup);
|
||||||
layerObjectGroup.add(konvaImageGroup);
|
mapping.konvaObjectGroup.add(konvaImageGroup);
|
||||||
|
|
||||||
|
entry = mapping.addEntry({ id: imageObject.id, type: 'image', konvaGroup: konvaImageGroup, konvaImage: null });
|
||||||
getImageDTO(imageObject.image.name).then((imageDTO) => {
|
getImageDTO(imageObject.image.name).then((imageDTO) => {
|
||||||
if (!imageDTO) {
|
if (!imageDTO) {
|
||||||
placeholder.onError();
|
placeholder.onError();
|
||||||
@ -173,6 +202,7 @@ export const createImageObjectGroup = async (
|
|||||||
});
|
});
|
||||||
placeholder.onLoaded();
|
placeholder.onLoaded();
|
||||||
konvaImageGroup.add(konvaImage);
|
konvaImageGroup.add(konvaImage);
|
||||||
|
entry.konvaImage = konvaImage;
|
||||||
};
|
};
|
||||||
imageEl.onerror = () => {
|
imageEl.onerror = () => {
|
||||||
placeholder.onError();
|
placeholder.onError();
|
||||||
@ -180,7 +210,7 @@ export const createImageObjectGroup = async (
|
|||||||
imageEl.id = imageObject.id;
|
imageEl.id = imageObject.id;
|
||||||
imageEl.src = imageDTO.image_url;
|
imageEl.src = imageDTO.image_url;
|
||||||
});
|
});
|
||||||
return konvaImageGroup;
|
return entry;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import type { EntityToKonvaMap, EntityToKonvaMapping } from 'features/controlLayers/konva/konvaMap';
|
||||||
import {
|
import {
|
||||||
RASTER_LAYER_BRUSH_LINE_NAME,
|
RASTER_LAYER_BRUSH_LINE_NAME,
|
||||||
RASTER_LAYER_ERASER_LINE_NAME,
|
RASTER_LAYER_ERASER_LINE_NAME,
|
||||||
@ -9,11 +10,11 @@ import {
|
|||||||
import {
|
import {
|
||||||
createImageObjectGroup,
|
createImageObjectGroup,
|
||||||
createObjectGroup,
|
createObjectGroup,
|
||||||
createRectShape,
|
|
||||||
getBrushLine,
|
getBrushLine,
|
||||||
getEraserLine,
|
getEraserLine,
|
||||||
|
getRectShape,
|
||||||
} from 'features/controlLayers/konva/renderers/objects';
|
} from 'features/controlLayers/konva/renderers/objects';
|
||||||
import { mapId, selectRasterObjects } from 'features/controlLayers/konva/util';
|
import { mapId } from 'features/controlLayers/konva/util';
|
||||||
import type { CanvasEntity, LayerEntity, PosChangedArg, Tool } from 'features/controlLayers/store/types';
|
import type { CanvasEntity, LayerEntity, PosChangedArg, Tool } from 'features/controlLayers/store/types';
|
||||||
import Konva from 'konva';
|
import Konva from 'konva';
|
||||||
|
|
||||||
@ -27,11 +28,16 @@ import Konva from 'konva';
|
|||||||
* @param layerState The raster layer state
|
* @param layerState The raster layer state
|
||||||
* @param onPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
const createRasterLayer = (
|
const getLayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
layerMap: EntityToKonvaMap,
|
||||||
layerState: LayerEntity,
|
layerState: LayerEntity,
|
||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): Konva.Layer => {
|
): EntityToKonvaMapping => {
|
||||||
|
let mapping = layerMap.getMapping(layerState.id);
|
||||||
|
if (mapping) {
|
||||||
|
return mapping;
|
||||||
|
}
|
||||||
// This layer hasn't been added to the konva state yet
|
// This layer hasn't been added to the konva state yet
|
||||||
const konvaLayer = new Konva.Layer({
|
const konvaLayer = new Konva.Layer({
|
||||||
id: layerState.id,
|
id: layerState.id,
|
||||||
@ -48,9 +54,11 @@ const createRasterLayer = (
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const konvaObjectGroup = createObjectGroup(konvaLayer, RASTER_LAYER_OBJECT_GROUP_NAME);
|
||||||
|
konvaLayer.add(konvaObjectGroup);
|
||||||
stage.add(konvaLayer);
|
stage.add(konvaLayer);
|
||||||
|
mapping = layerMap.addMapping(layerState.id, konvaLayer, konvaObjectGroup);
|
||||||
return konvaLayer;
|
return mapping;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -62,63 +70,51 @@ const createRasterLayer = (
|
|||||||
*/
|
*/
|
||||||
export const renderRasterLayer = async (
|
export const renderRasterLayer = async (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
layerMap: EntityToKonvaMap,
|
||||||
layerState: LayerEntity,
|
layerState: LayerEntity,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
) => {
|
) => {
|
||||||
const konvaLayer =
|
const mapping = getLayer(stage, layerMap, layerState, onPosChanged);
|
||||||
stage.findOne<Konva.Layer>(`#${layerState.id}`) ?? createRasterLayer(stage, layerState, onPosChanged);
|
|
||||||
|
|
||||||
// Update the layer's position and listening state
|
// Update the layer's position and listening state
|
||||||
konvaLayer.setAttrs({
|
mapping.konvaLayer.setAttrs({
|
||||||
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
||||||
x: Math.floor(layerState.x),
|
x: Math.floor(layerState.x),
|
||||||
y: Math.floor(layerState.y),
|
y: Math.floor(layerState.y),
|
||||||
});
|
});
|
||||||
|
|
||||||
const konvaObjectGroup =
|
|
||||||
konvaLayer.findOne<Konva.Group>(`.${RASTER_LAYER_OBJECT_GROUP_NAME}`) ??
|
|
||||||
createObjectGroup(konvaLayer, RASTER_LAYER_OBJECT_GROUP_NAME);
|
|
||||||
|
|
||||||
const objectIds = layerState.objects.map(mapId);
|
const objectIds = layerState.objects.map(mapId);
|
||||||
// Destroy any objects that are no longer in the redux state
|
// Destroy any objects that are no longer in state
|
||||||
// TODO(psyche): `konvaObjectGroup.getChildren()` seems to return a stale array of children, but find is never stale.
|
for (const entry of mapping.getEntries()) {
|
||||||
// Should report upstream
|
if (!objectIds.includes(entry.id)) {
|
||||||
for (const objectNode of konvaObjectGroup.find(selectRasterObjects)) {
|
mapping.destroyEntry(entry.id);
|
||||||
if (!objectIds.includes(objectNode.id())) {
|
|
||||||
objectNode.destroy();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const obj of layerState.objects) {
|
for (const obj of layerState.objects) {
|
||||||
if (obj.type === 'brush_line') {
|
if (obj.type === 'brush_line') {
|
||||||
const konvaBrushLine = getBrushLine(obj, konvaObjectGroup, RASTER_LAYER_BRUSH_LINE_NAME);
|
const entry = getBrushLine(mapping, obj, RASTER_LAYER_BRUSH_LINE_NAME);
|
||||||
// Only update the points if they have changed.
|
// Only update the points if they have changed.
|
||||||
if (konvaBrushLine.points().length !== obj.points.length) {
|
if (entry.konvaLine.points().length !== obj.points.length) {
|
||||||
konvaBrushLine.points(obj.points);
|
entry.konvaLine.points(obj.points);
|
||||||
}
|
}
|
||||||
} else if (obj.type === 'eraser_line') {
|
} else if (obj.type === 'eraser_line') {
|
||||||
const konvaEraserLine =
|
const entry = getEraserLine(mapping, obj, RASTER_LAYER_ERASER_LINE_NAME);
|
||||||
konvaObjectGroup.findOne<Konva.Line>(`#${obj.id}`) ??
|
|
||||||
getEraserLine(obj, konvaObjectGroup, RASTER_LAYER_ERASER_LINE_NAME);
|
|
||||||
// Only update the points if they have changed.
|
// Only update the points if they have changed.
|
||||||
if (konvaEraserLine.points().length !== obj.points.length) {
|
if (entry.konvaLine.points().length !== obj.points.length) {
|
||||||
konvaEraserLine.points(obj.points);
|
entry.konvaLine.points(obj.points);
|
||||||
}
|
}
|
||||||
} else if (obj.type === 'rect_shape') {
|
} else if (obj.type === 'rect_shape') {
|
||||||
if (!konvaObjectGroup.findOne<Konva.Rect>(`#${obj.id}`)) {
|
getRectShape(mapping, obj, RASTER_LAYER_RECT_SHAPE_NAME);
|
||||||
createRectShape(obj, konvaObjectGroup, RASTER_LAYER_RECT_SHAPE_NAME);
|
|
||||||
}
|
|
||||||
} else if (obj.type === 'image') {
|
} else if (obj.type === 'image') {
|
||||||
if (!konvaObjectGroup.findOne<Konva.Group>(`#${obj.id}`)) {
|
createImageObjectGroup(mapping, obj, RASTER_LAYER_IMAGE_NAME);
|
||||||
createImageObjectGroup(obj, konvaObjectGroup, RASTER_LAYER_IMAGE_NAME);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only update layer visibility if it has changed.
|
// Only update layer visibility if it has changed.
|
||||||
if (konvaLayer.visible() !== layerState.isEnabled) {
|
if (mapping.konvaLayer.visible() !== layerState.isEnabled) {
|
||||||
konvaLayer.visible(layerState.isEnabled);
|
mapping.konvaLayer.visible(layerState.isEnabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
// const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(layerState, konvaLayer);
|
// const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(layerState, konvaLayer);
|
||||||
@ -139,22 +135,23 @@ export const renderRasterLayer = async (
|
|||||||
// bboxRect.visible(false);
|
// bboxRect.visible(false);
|
||||||
// }
|
// }
|
||||||
|
|
||||||
konvaObjectGroup.opacity(layerState.opacity);
|
mapping.konvaObjectGroup.opacity(layerState.opacity);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const renderLayers = (
|
export const renderLayers = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
layerMap: EntityToKonvaMap,
|
||||||
layers: LayerEntity[],
|
layers: LayerEntity[],
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): void => {
|
): void => {
|
||||||
// Destroy nonexistent layers
|
// Destroy nonexistent layers
|
||||||
for (const konvaLayer of stage.find<Konva.Layer>(`.${RASTER_LAYER_NAME}`)) {
|
for (const mapping of layerMap.getMappings()) {
|
||||||
if (!layers.find((l) => l.id === konvaLayer.id())) {
|
if (!layers.find((l) => l.id === mapping.id)) {
|
||||||
konvaLayer.destroy();
|
layerMap.destroyMapping(mapping.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const layer of layers) {
|
for (const layer of layers) {
|
||||||
renderRasterLayer(stage, layer, tool, onPosChanged);
|
renderRasterLayer(stage, layerMap, layer, tool, onPosChanged);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -4,6 +4,7 @@ import { logger } from 'app/logging/logger';
|
|||||||
import { $isDebugging } from 'app/store/nanostores/isDebugging';
|
import { $isDebugging } from 'app/store/nanostores/isDebugging';
|
||||||
import type { RootState } from 'app/store/store';
|
import type { RootState } from 'app/store/store';
|
||||||
import { setStageEventHandlers } from 'features/controlLayers/konva/events';
|
import { setStageEventHandlers } from 'features/controlLayers/konva/events';
|
||||||
|
import { EntityToKonvaMap } from 'features/controlLayers/konva/konvaMap';
|
||||||
import { renderBackgroundLayer } from 'features/controlLayers/konva/renderers/background';
|
import { renderBackgroundLayer } from 'features/controlLayers/konva/renderers/background';
|
||||||
import { updateBboxes } from 'features/controlLayers/konva/renderers/bbox';
|
import { updateBboxes } from 'features/controlLayers/konva/renderers/bbox';
|
||||||
import { renderControlAdapters } from 'features/controlLayers/konva/renderers/caLayer';
|
import { renderControlAdapters } from 'features/controlLayers/konva/renderers/caLayer';
|
||||||
@ -281,6 +282,10 @@ export const initializeRenderer = (
|
|||||||
// the entire state over when needed.
|
// the entire state over when needed.
|
||||||
const debouncedUpdateBboxes = debounce(updateBboxes, 300);
|
const debouncedUpdateBboxes = debounce(updateBboxes, 300);
|
||||||
|
|
||||||
|
const regionMap = new EntityToKonvaMap();
|
||||||
|
const layerMap = new EntityToKonvaMap();
|
||||||
|
const controlAdapterMap = new EntityToKonvaMap();
|
||||||
|
|
||||||
const renderCanvas = () => {
|
const renderCanvas = () => {
|
||||||
const { canvasV2 } = store.getState();
|
const { canvasV2 } = store.getState();
|
||||||
|
|
||||||
@ -298,7 +303,7 @@ export const initializeRenderer = (
|
|||||||
canvasV2.tool.selected !== prevCanvasV2.tool.selected
|
canvasV2.tool.selected !== prevCanvasV2.tool.selected
|
||||||
) {
|
) {
|
||||||
logIfDebugging('Rendering layers');
|
logIfDebugging('Rendering layers');
|
||||||
renderLayers(stage, canvasV2.layers, canvasV2.tool.selected, onPosChanged);
|
renderLayers(stage, layerMap, canvasV2.layers, canvasV2.tool.selected, onPosChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
@ -310,6 +315,7 @@ export const initializeRenderer = (
|
|||||||
logIfDebugging('Rendering regions');
|
logIfDebugging('Rendering regions');
|
||||||
renderRegions(
|
renderRegions(
|
||||||
stage,
|
stage,
|
||||||
|
regionMap,
|
||||||
canvasV2.regions,
|
canvasV2.regions,
|
||||||
canvasV2.settings.maskOpacity,
|
canvasV2.settings.maskOpacity,
|
||||||
canvasV2.tool.selected,
|
canvasV2.tool.selected,
|
||||||
@ -320,7 +326,7 @@ export const initializeRenderer = (
|
|||||||
|
|
||||||
if (isFirstRender || canvasV2.controlAdapters !== prevCanvasV2.controlAdapters) {
|
if (isFirstRender || canvasV2.controlAdapters !== prevCanvasV2.controlAdapters) {
|
||||||
logIfDebugging('Rendering control adapters');
|
logIfDebugging('Rendering control adapters');
|
||||||
renderControlAdapters(stage, canvasV2.controlAdapters, getImageDTO);
|
renderControlAdapters(stage, controlAdapterMap, canvasV2.controlAdapters, getImageDTO);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isFirstRender || canvasV2.document !== prevCanvasV2.document) {
|
if (isFirstRender || canvasV2.document !== prevCanvasV2.document) {
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
import { rgbColorToString } from 'common/util/colorCodeTransformers';
|
import { rgbColorToString } from 'common/util/colorCodeTransformers';
|
||||||
import { BBOX_SELECTED_STROKE } from 'features/controlLayers/konva/constants';
|
import type { EntityToKonvaMap, EntityToKonvaMapping } from 'features/controlLayers/konva/konvaMap';
|
||||||
import {
|
import {
|
||||||
COMPOSITING_RECT_NAME,
|
COMPOSITING_RECT_NAME,
|
||||||
LAYER_BBOX_NAME,
|
|
||||||
RG_LAYER_BRUSH_LINE_NAME,
|
RG_LAYER_BRUSH_LINE_NAME,
|
||||||
RG_LAYER_ERASER_LINE_NAME,
|
RG_LAYER_ERASER_LINE_NAME,
|
||||||
RG_LAYER_NAME,
|
RG_LAYER_NAME,
|
||||||
@ -11,13 +10,12 @@ import {
|
|||||||
} from 'features/controlLayers/konva/naming';
|
} from 'features/controlLayers/konva/naming';
|
||||||
import { getLayerBboxFast } from 'features/controlLayers/konva/renderers/bbox';
|
import { getLayerBboxFast } from 'features/controlLayers/konva/renderers/bbox';
|
||||||
import {
|
import {
|
||||||
createBboxRect,
|
|
||||||
createObjectGroup,
|
createObjectGroup,
|
||||||
createRectShape,
|
|
||||||
getBrushLine,
|
getBrushLine,
|
||||||
getEraserLine,
|
getEraserLine,
|
||||||
|
getRectShape,
|
||||||
} from 'features/controlLayers/konva/renderers/objects';
|
} from 'features/controlLayers/konva/renderers/objects';
|
||||||
import { mapId, selectVectorMaskObjects } from 'features/controlLayers/konva/util';
|
import { mapId } from 'features/controlLayers/konva/util';
|
||||||
import type {
|
import type {
|
||||||
CanvasEntity,
|
CanvasEntity,
|
||||||
CanvasEntityIdentifier,
|
CanvasEntityIdentifier,
|
||||||
@ -47,17 +45,22 @@ const createCompositingRect = (konvaLayer: Konva.Layer): Konva.Rect => {
|
|||||||
/**
|
/**
|
||||||
* Creates a regional guidance layer.
|
* Creates a regional guidance layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param rg The regional guidance layer state
|
* @param region The regional guidance layer state
|
||||||
* @param onLayerPosChanged Callback for when the layer's position changes
|
* @param onLayerPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
const createRGLayer = (
|
const getRegion = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
rg: RegionEntity,
|
regionMap: EntityToKonvaMap,
|
||||||
|
region: RegionEntity,
|
||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): Konva.Layer => {
|
): EntityToKonvaMapping => {
|
||||||
|
let mapping = regionMap.getMapping(region.id);
|
||||||
|
if (mapping) {
|
||||||
|
return mapping;
|
||||||
|
}
|
||||||
// This layer hasn't been added to the konva state yet
|
// This layer hasn't been added to the konva state yet
|
||||||
const konvaLayer = new Konva.Layer({
|
const konvaLayer = new Konva.Layer({
|
||||||
id: rg.id,
|
id: region.id,
|
||||||
name: RG_LAYER_NAME,
|
name: RG_LAYER_NAME,
|
||||||
draggable: true,
|
draggable: true,
|
||||||
dragDistance: 0,
|
dragDistance: 0,
|
||||||
@ -67,117 +70,114 @@ const createRGLayer = (
|
|||||||
// the position - we do not need to call this on the `dragmove` event.
|
// the position - we do not need to call this on the `dragmove` event.
|
||||||
if (onPosChanged) {
|
if (onPosChanged) {
|
||||||
konvaLayer.on('dragend', function (e) {
|
konvaLayer.on('dragend', function (e) {
|
||||||
onPosChanged({ id: rg.id, x: Math.floor(e.target.x()), y: Math.floor(e.target.y()) }, 'regional_guidance');
|
onPosChanged({ id: region.id, x: Math.floor(e.target.x()), y: Math.floor(e.target.y()) }, 'regional_guidance');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
stage.add(konvaLayer);
|
const konvaObjectGroup = createObjectGroup(konvaLayer, RG_LAYER_OBJECT_GROUP_NAME);
|
||||||
|
|
||||||
return konvaLayer;
|
konvaLayer.add(konvaObjectGroup);
|
||||||
|
stage.add(konvaLayer);
|
||||||
|
mapping = regionMap.addMapping(region.id, konvaLayer, konvaObjectGroup);
|
||||||
|
return mapping;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Renders a raster layer.
|
* Renders a raster layer.
|
||||||
* @param stage The konva stage
|
* @param stage The konva stage
|
||||||
* @param rg The regional guidance layer state
|
* @param region The regional guidance layer state
|
||||||
* @param globalMaskLayerOpacity The global mask layer opacity
|
* @param globalMaskLayerOpacity The global mask layer opacity
|
||||||
* @param tool The current tool
|
* @param tool The current tool
|
||||||
* @param onPosChanged Callback for when the layer's position changes
|
* @param onPosChanged Callback for when the layer's position changes
|
||||||
*/
|
*/
|
||||||
export const renderRGLayer = (
|
export const renderRGLayer = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
rg: RegionEntity,
|
regionMap: EntityToKonvaMap,
|
||||||
|
region: RegionEntity,
|
||||||
globalMaskLayerOpacity: number,
|
globalMaskLayerOpacity: number,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
selectedEntityIdentifier: CanvasEntityIdentifier | null,
|
selectedEntityIdentifier: CanvasEntityIdentifier | null,
|
||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): void => {
|
): void => {
|
||||||
const konvaLayer = stage.findOne<Konva.Layer>(`#${rg.id}`) ?? createRGLayer(stage, rg, onPosChanged);
|
const mapping = getRegion(stage, regionMap, region, onPosChanged);
|
||||||
|
|
||||||
// Update the layer's position and listening state
|
// Update the layer's position and listening state
|
||||||
konvaLayer.setAttrs({
|
mapping.konvaLayer.setAttrs({
|
||||||
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
listening: tool === 'move', // The layer only listens when using the move tool - otherwise the stage is handling mouse events
|
||||||
x: Math.floor(rg.x),
|
x: Math.floor(region.x),
|
||||||
y: Math.floor(rg.y),
|
y: Math.floor(region.y),
|
||||||
});
|
});
|
||||||
|
|
||||||
// Convert the color to a string, stripping the alpha - the object group will handle opacity.
|
// Convert the color to a string, stripping the alpha - the object group will handle opacity.
|
||||||
const rgbColor = rgbColorToString(rg.fill);
|
const rgbColor = rgbColorToString(region.fill);
|
||||||
|
|
||||||
const konvaObjectGroup =
|
|
||||||
konvaLayer.findOne<Konva.Group>(`.${RG_LAYER_OBJECT_GROUP_NAME}`) ??
|
|
||||||
createObjectGroup(konvaLayer, RG_LAYER_OBJECT_GROUP_NAME);
|
|
||||||
|
|
||||||
// We use caching to handle "global" layer opacity, but caching is expensive and we should only do it when required.
|
// We use caching to handle "global" layer opacity, but caching is expensive and we should only do it when required.
|
||||||
let groupNeedsCache = false;
|
let groupNeedsCache = false;
|
||||||
|
|
||||||
const objectIds = rg.objects.map(mapId);
|
const objectIds = region.objects.map(mapId);
|
||||||
// Destroy any objects that are no longer in the redux state
|
// Destroy any objects that are no longer in state
|
||||||
for (const objectNode of konvaObjectGroup.find(selectVectorMaskObjects)) {
|
for (const entry of mapping.getEntries()) {
|
||||||
if (!objectIds.includes(objectNode.id())) {
|
if (!objectIds.includes(entry.id)) {
|
||||||
objectNode.destroy();
|
mapping.destroyEntry(entry.id);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const obj of rg.objects) {
|
for (const obj of region.objects) {
|
||||||
if (obj.type === 'brush_line') {
|
if (obj.type === 'brush_line') {
|
||||||
const konvaBrushLine =
|
const entry = getBrushLine(mapping, obj, RG_LAYER_BRUSH_LINE_NAME);
|
||||||
stage.findOne<Konva.Line>(`#${obj.id}`) ?? getBrushLine(obj, konvaObjectGroup, RG_LAYER_BRUSH_LINE_NAME);
|
|
||||||
|
|
||||||
// Only update the points if they have changed. The point values are never mutated, they are only added to the
|
// Only update the points if they have changed. The point values are never mutated, they are only added to the
|
||||||
// array, so checking the length is sufficient to determine if we need to re-cache.
|
// array, so checking the length is sufficient to determine if we need to re-cache.
|
||||||
if (konvaBrushLine.points().length !== obj.points.length) {
|
if (entry.konvaLine.points().length !== obj.points.length) {
|
||||||
konvaBrushLine.points(obj.points);
|
entry.konvaLine.points(obj.points);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
// Only update the color if it has changed.
|
// Only update the color if it has changed.
|
||||||
if (konvaBrushLine.stroke() !== rgbColor) {
|
if (entry.konvaLine.stroke() !== rgbColor) {
|
||||||
konvaBrushLine.stroke(rgbColor);
|
entry.konvaLine.stroke(rgbColor);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
} else if (obj.type === 'eraser_line') {
|
} else if (obj.type === 'eraser_line') {
|
||||||
const konvaEraserLine =
|
const entry = getEraserLine(mapping, obj, RG_LAYER_ERASER_LINE_NAME);
|
||||||
stage.findOne<Konva.Line>(`#${obj.id}`) ?? getEraserLine(obj, konvaObjectGroup, RG_LAYER_ERASER_LINE_NAME);
|
|
||||||
|
|
||||||
// Only update the points if they have changed. The point values are never mutated, they are only added to the
|
// Only update the points if they have changed. The point values are never mutated, they are only added to the
|
||||||
// array, so checking the length is sufficient to determine if we need to re-cache.
|
// array, so checking the length is sufficient to determine if we need to re-cache.
|
||||||
if (konvaEraserLine.points().length !== obj.points.length) {
|
if (entry.konvaLine.points().length !== obj.points.length) {
|
||||||
konvaEraserLine.points(obj.points);
|
entry.konvaLine.points(obj.points);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
// Only update the color if it has changed.
|
// Only update the color if it has changed.
|
||||||
if (konvaEraserLine.stroke() !== rgbColor) {
|
if (entry.konvaLine.stroke() !== rgbColor) {
|
||||||
konvaEraserLine.stroke(rgbColor);
|
entry.konvaLine.stroke(rgbColor);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
} else if (obj.type === 'rect_shape') {
|
} else if (obj.type === 'rect_shape') {
|
||||||
const konvaRectShape =
|
const entry = getRectShape(mapping, obj, RG_LAYER_RECT_SHAPE_NAME);
|
||||||
stage.findOne<Konva.Rect>(`#${obj.id}`) ?? createRectShape(obj, konvaObjectGroup, RG_LAYER_RECT_SHAPE_NAME);
|
|
||||||
|
|
||||||
// Only update the color if it has changed.
|
// Only update the color if it has changed.
|
||||||
if (konvaRectShape.fill() !== rgbColor) {
|
if (entry.konvaRect.fill() !== rgbColor) {
|
||||||
konvaRectShape.fill(rgbColor);
|
entry.konvaRect.fill(rgbColor);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only update layer visibility if it has changed.
|
// Only update layer visibility if it has changed.
|
||||||
if (konvaLayer.visible() !== rg.isEnabled) {
|
if (mapping.konvaLayer.visible() !== region.isEnabled) {
|
||||||
konvaLayer.visible(rg.isEnabled);
|
mapping.konvaLayer.visible(region.isEnabled);
|
||||||
groupNeedsCache = true;
|
groupNeedsCache = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (konvaObjectGroup.getChildren().length === 0) {
|
if (mapping.konvaObjectGroup.getChildren().length === 0) {
|
||||||
// No objects - clear the cache to reset the previous pixel data
|
// No objects - clear the cache to reset the previous pixel data
|
||||||
konvaObjectGroup.clearCache();
|
mapping.konvaObjectGroup.clearCache();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const compositingRect =
|
const compositingRect =
|
||||||
konvaLayer.findOne<Konva.Rect>(`.${COMPOSITING_RECT_NAME}`) ?? createCompositingRect(konvaLayer);
|
mapping.konvaLayer.findOne<Konva.Rect>(`.${COMPOSITING_RECT_NAME}`) ?? createCompositingRect(mapping.konvaLayer);
|
||||||
const isSelected = selectedEntityIdentifier?.id === rg.id;
|
const isSelected = selectedEntityIdentifier?.id === region.id;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||||
@ -192,54 +192,56 @@ export const renderRGLayer = (
|
|||||||
*/
|
*/
|
||||||
if (isSelected && tool !== 'move') {
|
if (isSelected && tool !== 'move') {
|
||||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||||
if (konvaObjectGroup.isCached()) {
|
if (mapping.konvaObjectGroup.isCached()) {
|
||||||
konvaObjectGroup.clearCache();
|
mapping.konvaObjectGroup.clearCache();
|
||||||
}
|
}
|
||||||
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||||
konvaObjectGroup.opacity(1);
|
mapping.konvaObjectGroup.opacity(1);
|
||||||
|
|
||||||
compositingRect.setAttrs({
|
compositingRect.setAttrs({
|
||||||
// The rect should be the size of the layer - use the fast method if we don't have a pixel-perfect bbox already
|
// The rect should be the size of the layer - use the fast method if we don't have a pixel-perfect bbox already
|
||||||
...(!rg.bboxNeedsUpdate && rg.bbox ? rg.bbox : getLayerBboxFast(konvaLayer)),
|
...(!region.bboxNeedsUpdate && region.bbox ? region.bbox : getLayerBboxFast(mapping.konvaLayer)),
|
||||||
fill: rgbColor,
|
fill: rgbColor,
|
||||||
opacity: globalMaskLayerOpacity,
|
opacity: globalMaskLayerOpacity,
|
||||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||||
globalCompositeOperation: 'source-in',
|
globalCompositeOperation: 'source-in',
|
||||||
visible: true,
|
visible: true,
|
||||||
// This rect must always be on top of all other shapes
|
// This rect must always be on top of all other shapes
|
||||||
zIndex: konvaObjectGroup.getChildren().length,
|
zIndex: mapping.konvaObjectGroup.getChildren().length,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
// The compositing rect should only be shown when the layer is selected.
|
// The compositing rect should only be shown when the layer is selected.
|
||||||
compositingRect.visible(false);
|
compositingRect.visible(false);
|
||||||
// Cache only if needed - or if we are on this code path and _don't_ have a cache
|
// Cache only if needed - or if we are on this code path and _don't_ have a cache
|
||||||
if (groupNeedsCache || !konvaObjectGroup.isCached()) {
|
if (groupNeedsCache || !mapping.konvaObjectGroup.isCached()) {
|
||||||
konvaObjectGroup.cache();
|
mapping.konvaObjectGroup.cache();
|
||||||
}
|
}
|
||||||
// Updating group opacity does not require re-caching
|
// Updating group opacity does not require re-caching
|
||||||
konvaObjectGroup.opacity(globalMaskLayerOpacity);
|
mapping.konvaObjectGroup.opacity(globalMaskLayerOpacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
const bboxRect = konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(rg, konvaLayer);
|
// const bboxRect =
|
||||||
|
// regionMap.konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(rg, regionMap.konvaLayer);
|
||||||
|
|
||||||
if (rg.bbox) {
|
// if (rg.bbox) {
|
||||||
const active = !rg.bboxNeedsUpdate && isSelected && tool === 'move';
|
// const active = !rg.bboxNeedsUpdate && isSelected && tool === 'move';
|
||||||
bboxRect.setAttrs({
|
// bboxRect.setAttrs({
|
||||||
visible: active,
|
// visible: active,
|
||||||
listening: active,
|
// listening: active,
|
||||||
x: rg.bbox.x,
|
// x: rg.bbox.x,
|
||||||
y: rg.bbox.y,
|
// y: rg.bbox.y,
|
||||||
width: rg.bbox.width,
|
// width: rg.bbox.width,
|
||||||
height: rg.bbox.height,
|
// height: rg.bbox.height,
|
||||||
stroke: isSelected ? BBOX_SELECTED_STROKE : '',
|
// stroke: isSelected ? BBOX_SELECTED_STROKE : '',
|
||||||
});
|
// });
|
||||||
} else {
|
// } else {
|
||||||
bboxRect.visible(false);
|
// bboxRect.visible(false);
|
||||||
}
|
// }
|
||||||
};
|
};
|
||||||
|
|
||||||
export const renderRegions = (
|
export const renderRegions = (
|
||||||
stage: Konva.Stage,
|
stage: Konva.Stage,
|
||||||
|
regionMap: EntityToKonvaMap,
|
||||||
regions: RegionEntity[],
|
regions: RegionEntity[],
|
||||||
maskOpacity: number,
|
maskOpacity: number,
|
||||||
tool: Tool,
|
tool: Tool,
|
||||||
@ -247,12 +249,12 @@ export const renderRegions = (
|
|||||||
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
onPosChanged?: (arg: PosChangedArg, entityType: CanvasEntity['type']) => void
|
||||||
): void => {
|
): void => {
|
||||||
// Destroy nonexistent layers
|
// Destroy nonexistent layers
|
||||||
for (const konvaLayer of stage.find<Konva.Layer>(`.${RG_LAYER_NAME}`)) {
|
for (const mapping of regionMap.getMappings()) {
|
||||||
if (!regions.find((rg) => rg.id === konvaLayer.id())) {
|
if (!regions.find((rg) => rg.id === mapping.id)) {
|
||||||
konvaLayer.destroy();
|
regionMap.destroyMapping(mapping.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const rg of regions) {
|
for (const rg of regions) {
|
||||||
renderRGLayer(stage, rg, maskOpacity, tool, selectedEntityIdentifier, onPosChanged);
|
renderRGLayer(stage, regionMap, rg, maskOpacity, tool, selectedEntityIdentifier, onPosChanged);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user