mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
feat(ui): staging area works more better
This commit is contained in:
parent
07b72c3d70
commit
eec3c3b884
@ -69,7 +69,7 @@ export const addStagingListeners = (startAppListening: AppStartListening) => {
|
||||
if (!layer) {
|
||||
// We need to create a new layer to add the accepted image
|
||||
api.dispatch(layerAdded());
|
||||
layer = layers.entities[0];
|
||||
layer = api.getState().canvasV2.layers.entities[0];
|
||||
}
|
||||
|
||||
assert(layer, 'No layer found to stage image');
|
||||
|
@ -25,6 +25,9 @@ export const addEnqueueRequestedLinear = (startAppListening: AppStartListening)
|
||||
assert(model, 'No model found in state');
|
||||
const base = model.base;
|
||||
|
||||
manager.getInpaintMaskImage({ bbox: state.canvasV2.bbox, preview: true });
|
||||
manager.getImageSourceImage({ bbox: state.canvasV2.bbox, preview: true });
|
||||
|
||||
if (base === 'sdxl') {
|
||||
graph = await buildSDXLGraph(state, manager);
|
||||
} else if (base === 'sd-1' || base === 'sd-2') {
|
||||
|
@ -61,7 +61,6 @@ export const StageComponent = memo(({ asPreview = false }: Props) => {
|
||||
bottom={0}
|
||||
left={0}
|
||||
ref={containerRef}
|
||||
tabIndex={-1}
|
||||
borderRadius="base"
|
||||
border={1}
|
||||
borderStyle="solid"
|
||||
|
@ -88,12 +88,6 @@ type Util = {
|
||||
image_category: ImageCategory,
|
||||
is_intermediate: boolean
|
||||
) => Promise<ImageDTO>;
|
||||
getRegionMaskImage: (arg: { id: string; bbox?: Rect; preview?: boolean }) => Promise<ImageDTO>;
|
||||
getInpaintMaskImage: (arg: { bbox?: Rect; preview?: boolean }) => Promise<ImageDTO>;
|
||||
getImageSourceImage: (arg: { bbox?: Rect; preview?: boolean }) => Promise<ImageDTO>;
|
||||
getMaskLayerClone: (arg: { id: string }) => Konva.Layer;
|
||||
getCompositeLayerStageClone: () => Konva.Stage;
|
||||
getGenerationMode: () => GenerationMode;
|
||||
};
|
||||
|
||||
const $nodeManager = atom<KonvaNodeManager | null>(null);
|
||||
@ -131,12 +125,6 @@ export class KonvaNodeManager {
|
||||
this.util = {
|
||||
getImageDTO,
|
||||
uploadImage,
|
||||
getRegionMaskImage: this._getRegionMaskImage.bind(this),
|
||||
getInpaintMaskImage: this._getInpaintMaskImage.bind(this),
|
||||
getImageSourceImage: this._getImageSourceImage.bind(this),
|
||||
getMaskLayerClone: this._getMaskLayerClone.bind(this),
|
||||
getCompositeLayerStageClone: this._getCompositeLayerStageClone.bind(this),
|
||||
getGenerationMode: this._getGenerationMode.bind(this),
|
||||
};
|
||||
|
||||
this.preview = new CanvasPreview(
|
||||
@ -310,9 +298,7 @@ export class KonvaNodeManager {
|
||||
this.renderDocumentSizeOverlay();
|
||||
}
|
||||
|
||||
_getMaskLayerClone(): Konva.Layer {
|
||||
assert(this.inpaintMask, 'Inpaint mask layer has not been set');
|
||||
|
||||
getInpaintMaskLayerClone(): Konva.Layer {
|
||||
const layerClone = this.inpaintMask.konvaLayer.clone();
|
||||
const objectGroupClone = this.inpaintMask.konvaObjectGroup.clone();
|
||||
|
||||
@ -325,7 +311,25 @@ export class KonvaNodeManager {
|
||||
return layerClone;
|
||||
}
|
||||
|
||||
_getCompositeLayerStageClone(): Konva.Stage {
|
||||
getRegionMaskLayerClone(arg: { id: string }): Konva.Layer {
|
||||
const { id } = arg;
|
||||
|
||||
const canvasRegion = this.regions.get(id);
|
||||
assert(canvasRegion, `Canvas region with id ${id} not found`);
|
||||
|
||||
const layerClone = canvasRegion.konvaLayer.clone();
|
||||
const objectGroupClone = canvasRegion.konvaObjectGroup.clone();
|
||||
|
||||
layerClone.destroyChildren();
|
||||
layerClone.add(objectGroupClone);
|
||||
|
||||
objectGroupClone.opacity(1);
|
||||
objectGroupClone.cache();
|
||||
|
||||
return layerClone;
|
||||
}
|
||||
|
||||
getCompositeLayerStageClone(): Konva.Stage {
|
||||
const layersState = this.stateApi.getLayersState();
|
||||
|
||||
const stageClone = this.stage.clone();
|
||||
@ -357,12 +361,12 @@ export class KonvaNodeManager {
|
||||
return stageClone;
|
||||
}
|
||||
|
||||
_getGenerationMode(): GenerationMode {
|
||||
getGenerationMode(): GenerationMode {
|
||||
const { x, y, width, height } = this.stateApi.getBbox();
|
||||
const inpaintMaskLayer = this.util.getMaskLayerClone({ id: 'inpaint_mask' });
|
||||
const inpaintMaskLayer = this.getInpaintMaskLayerClone();
|
||||
const inpaintMaskImageData = konvaNodeToImageData(inpaintMaskLayer, { x, y, width, height });
|
||||
const inpaintMaskTransparency = getImageDataTransparency(inpaintMaskImageData);
|
||||
const compositeLayer = this.util.getCompositeLayerStageClone();
|
||||
const compositeLayer = this.getCompositeLayerStageClone();
|
||||
const compositeLayerImageData = konvaNodeToImageData(compositeLayer, { x, y, width, height });
|
||||
const compositeLayerTransparency = getImageDataTransparency(compositeLayerImageData);
|
||||
if (compositeLayerTransparency.isPartiallyTransparent) {
|
||||
@ -378,7 +382,7 @@ export class KonvaNodeManager {
|
||||
}
|
||||
}
|
||||
|
||||
async _getRegionMaskImage(arg: { id: string; bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
async getRegionMaskImage(arg: { id: string; bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
const { id, bbox, preview = false } = arg;
|
||||
const region = this.stateApi.getRegionsState().entities.find((entity) => entity.id === id);
|
||||
assert(region, `Region entity state with id ${id} not found`);
|
||||
@ -390,7 +394,7 @@ export class KonvaNodeManager {
|
||||
// }
|
||||
// }
|
||||
|
||||
const layerClone = this.util.getMaskLayerClone({ id });
|
||||
const layerClone = this.getRegionMaskLayerClone({ id });
|
||||
const blob = await konvaNodeToBlob(layerClone, bbox);
|
||||
|
||||
if (preview) {
|
||||
@ -404,9 +408,9 @@ export class KonvaNodeManager {
|
||||
return imageDTO;
|
||||
}
|
||||
|
||||
async _getInpaintMaskImage(arg: { bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
async getInpaintMaskImage(arg: { bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
const { bbox, preview = false } = arg;
|
||||
const inpaintMask = this.stateApi.getInpaintMaskState();
|
||||
// const inpaintMask = this.stateApi.getInpaintMaskState();
|
||||
|
||||
// if (inpaintMask.imageCache) {
|
||||
// const imageDTO = await this.util.getImageDTO(inpaintMask.imageCache.name);
|
||||
@ -415,7 +419,7 @@ export class KonvaNodeManager {
|
||||
// }
|
||||
// }
|
||||
|
||||
const layerClone = this.util.getMaskLayerClone({ id: inpaintMask.id });
|
||||
const layerClone = this.getInpaintMaskLayerClone();
|
||||
const blob = await konvaNodeToBlob(layerClone, bbox);
|
||||
|
||||
if (preview) {
|
||||
@ -429,7 +433,7 @@ export class KonvaNodeManager {
|
||||
return imageDTO;
|
||||
}
|
||||
|
||||
async _getImageSourceImage(arg: { bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
async getImageSourceImage(arg: { bbox?: Rect; preview?: boolean }): Promise<ImageDTO> {
|
||||
const { bbox, preview = false } = arg;
|
||||
// const { imageCache } = this.stateApi.getLayersState();
|
||||
|
||||
@ -440,7 +444,7 @@ export class KonvaNodeManager {
|
||||
// }
|
||||
// }
|
||||
|
||||
const stageClone = this.util.getCompositeLayerStageClone();
|
||||
const stageClone = this.getCompositeLayerStageClone();
|
||||
|
||||
const blob = await konvaNodeToBlob(stageClone, bbox);
|
||||
|
||||
|
@ -67,6 +67,7 @@ export class CanvasInpaintMask {
|
||||
// Destroy any objects that are no longer in state
|
||||
for (const object of this.objects.values()) {
|
||||
if (!objectIds.includes(object.id)) {
|
||||
this.objects.delete(object.id);
|
||||
object.destroy();
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
@ -80,7 +81,7 @@ export class CanvasInpaintMask {
|
||||
if (!brushLine) {
|
||||
brushLine = new KonvaBrushLine({ brushLine: obj });
|
||||
this.objects.set(brushLine.id, brushLine);
|
||||
this.konvaLayer.add(brushLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(brushLine.konvaLineGroup);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
|
||||
@ -95,7 +96,7 @@ export class CanvasInpaintMask {
|
||||
if (!eraserLine) {
|
||||
eraserLine = new KonvaEraserLine({ eraserLine: obj });
|
||||
this.objects.set(eraserLine.id, eraserLine);
|
||||
this.konvaLayer.add(eraserLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(eraserLine.konvaLineGroup);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
|
||||
@ -110,7 +111,7 @@ export class CanvasInpaintMask {
|
||||
if (!rect) {
|
||||
rect = new KonvaRect({ rectShape: obj });
|
||||
this.objects.set(rect.id, rect);
|
||||
this.konvaLayer.add(rect.konvaRect);
|
||||
this.konvaObjectGroup.add(rect.konvaRect);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
}
|
||||
@ -128,50 +129,72 @@ export class CanvasInpaintMask {
|
||||
return;
|
||||
}
|
||||
|
||||
const isSelected = selectedEntityIdentifier?.id === inpaintMaskState.id;
|
||||
|
||||
/**
|
||||
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||
* shapes to render as a "raster" layer with all pixels drawn at the same color and opacity.
|
||||
*
|
||||
* Without this special handling, each shape is drawn individually with the given opacity, atop the other shapes. The
|
||||
* effect is like if you have a Photoshop Group consisting of many shapes, each of which has the given opacity.
|
||||
* Overlapping shapes will have their colors blended together, and the final color is the result of all the shapes.
|
||||
*
|
||||
* Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
||||
* a single raster image, and _then_ applied the 50% opacity.
|
||||
*/
|
||||
if (isSelected && selectedTool !== 'move') {
|
||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
if (this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.clearCache();
|
||||
}
|
||||
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
this.konvaObjectGroup.opacity(1);
|
||||
|
||||
this.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
|
||||
...(!inpaintMaskState.bboxNeedsUpdate && inpaintMaskState.bbox
|
||||
? inpaintMaskState.bbox
|
||||
: getLayerBboxFast(this.konvaLayer)),
|
||||
fill: rgbColor,
|
||||
opacity: maskOpacity,
|
||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
globalCompositeOperation: 'source-in',
|
||||
visible: true,
|
||||
// This rect must always be on top of all other shapes
|
||||
zIndex: this.objects.size + 1,
|
||||
});
|
||||
} else {
|
||||
// The compositing rect should only be shown when the layer is selected.
|
||||
this.compositingRect.visible(false);
|
||||
// Cache only if needed - or if we are on this code path and _don't_ have a cache
|
||||
if (groupNeedsCache || !this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.cache();
|
||||
}
|
||||
// Updating group opacity does not require re-caching
|
||||
this.konvaObjectGroup.opacity(maskOpacity);
|
||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
if (this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.clearCache();
|
||||
}
|
||||
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
this.konvaObjectGroup.opacity(1);
|
||||
|
||||
this.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
|
||||
...(!inpaintMaskState.bboxNeedsUpdate && inpaintMaskState.bbox
|
||||
? inpaintMaskState.bbox
|
||||
: getLayerBboxFast(this.konvaLayer)),
|
||||
fill: rgbColor,
|
||||
opacity: maskOpacity,
|
||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
globalCompositeOperation: 'source-in',
|
||||
visible: true,
|
||||
// This rect must always be on top of all other shapes
|
||||
zIndex: this.objects.size + 1,
|
||||
});
|
||||
|
||||
// const isSelected = selectedEntityIdentifier?.id === inpaintMaskState.id;
|
||||
|
||||
// /**
|
||||
// * When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||
// * shapes to render as a "raster" layer with all pixels drawn at the same color and opacity.
|
||||
// *
|
||||
// * Without this special handling, each shape is drawn individually with the given opacity, atop the other shapes. The
|
||||
// * effect is like if you have a Photoshop Group consisting of many shapes, each of which has the given opacity.
|
||||
// * Overlapping shapes will have their colors blended together, and the final color is the result of all the shapes.
|
||||
// *
|
||||
// * Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
||||
// * a single raster image, and _then_ applied the 50% opacity.
|
||||
// */
|
||||
// if (isSelected && selectedTool !== 'move') {
|
||||
// // We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
// if (this.konvaObjectGroup.isCached()) {
|
||||
// this.konvaObjectGroup.clearCache();
|
||||
// }
|
||||
// // The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
// this.konvaObjectGroup.opacity(1);
|
||||
|
||||
// this.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
|
||||
// ...(!inpaintMaskState.bboxNeedsUpdate && inpaintMaskState.bbox
|
||||
// ? inpaintMaskState.bbox
|
||||
// : getLayerBboxFast(this.konvaLayer)),
|
||||
// fill: rgbColor,
|
||||
// opacity: maskOpacity,
|
||||
// // Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
// globalCompositeOperation: 'source-in',
|
||||
// visible: true,
|
||||
// // This rect must always be on top of all other shapes
|
||||
// zIndex: this.objects.size + 1,
|
||||
// });
|
||||
// } else {
|
||||
// // The compositing rect should only be shown when the layer is selected.
|
||||
// this.compositingRect.visible(false);
|
||||
// // Cache only if needed - or if we are on this code path and _don't_ have a cache
|
||||
// if (groupNeedsCache || !this.konvaObjectGroup.isCached()) {
|
||||
// this.konvaObjectGroup.cache();
|
||||
// }
|
||||
// // Updating group opacity does not require re-caching
|
||||
// this.konvaObjectGroup.opacity(maskOpacity);
|
||||
// }
|
||||
|
||||
// const bboxRect =
|
||||
// regionMap.konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(rg, regionMap.konvaLayer);
|
||||
|
@ -52,6 +52,7 @@ export class CanvasLayer {
|
||||
// Destroy any objects that are no longer in state
|
||||
for (const object of this.objects.values()) {
|
||||
if (!objectIds.includes(object.id)) {
|
||||
this.objects.delete(object.id);
|
||||
object.destroy();
|
||||
}
|
||||
}
|
||||
@ -64,7 +65,7 @@ export class CanvasLayer {
|
||||
if (!brushLine) {
|
||||
brushLine = new KonvaBrushLine({ brushLine: obj });
|
||||
this.objects.set(brushLine.id, brushLine);
|
||||
this.konvaLayer.add(brushLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(brushLine.konvaLineGroup);
|
||||
}
|
||||
if (obj.points.length !== brushLine.konvaLine.points().length) {
|
||||
brushLine.konvaLine.points(obj.points);
|
||||
@ -76,7 +77,7 @@ export class CanvasLayer {
|
||||
if (!eraserLine) {
|
||||
eraserLine = new KonvaEraserLine({ eraserLine: obj });
|
||||
this.objects.set(eraserLine.id, eraserLine);
|
||||
this.konvaLayer.add(eraserLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(eraserLine.konvaLineGroup);
|
||||
}
|
||||
if (obj.points.length !== eraserLine.konvaLine.points().length) {
|
||||
eraserLine.konvaLine.points(obj.points);
|
||||
@ -88,7 +89,7 @@ export class CanvasLayer {
|
||||
if (!rect) {
|
||||
rect = new KonvaRect({ rectShape: obj });
|
||||
this.objects.set(rect.id, rect);
|
||||
this.konvaLayer.add(rect.konvaRect);
|
||||
this.konvaObjectGroup.add(rect.konvaRect);
|
||||
}
|
||||
} else if (obj.type === 'image') {
|
||||
let image = this.objects.get(obj.id);
|
||||
@ -97,7 +98,7 @@ export class CanvasLayer {
|
||||
if (!image) {
|
||||
image = await new KonvaImage({ imageObject: obj });
|
||||
this.objects.set(image.id, image);
|
||||
this.konvaLayer.add(image.konvaImageGroup);
|
||||
this.konvaObjectGroup.add(image.konvaImageGroup);
|
||||
}
|
||||
if (image.imageName !== obj.image.name) {
|
||||
image.updateImageSource(obj.image.name);
|
||||
|
@ -1,7 +1,7 @@
|
||||
import { rgbaColorToString } from 'common/util/colorCodeTransformers';
|
||||
import { getLayerBboxId, LAYER_BBOX_NAME } from 'features/controlLayers/konva/naming';
|
||||
import type { BrushLine, CanvasEntity, EraserLine, ImageObject, RectShape } from 'features/controlLayers/store/types';
|
||||
import { DEFAULT_RGBA_COLOR } from 'features/controlLayers/store/types';
|
||||
import { RGBA_RED } from 'features/controlLayers/store/types';
|
||||
import { t } from 'i18next';
|
||||
import Konva from 'konva';
|
||||
import { getImageDTO as defaultGetImageDTO } from 'services/api/endpoints/images';
|
||||
@ -77,7 +77,7 @@ export class KonvaEraserLine {
|
||||
lineCap: 'round',
|
||||
lineJoin: 'round',
|
||||
globalCompositeOperation: 'destination-out',
|
||||
stroke: rgbaColorToString(DEFAULT_RGBA_COLOR),
|
||||
stroke: rgbaColorToString(RGBA_RED),
|
||||
});
|
||||
this.konvaLineGroup.add(this.konvaLine);
|
||||
}
|
||||
|
@ -215,6 +215,7 @@ export class CanvasTool {
|
||||
strokeEnabled: false,
|
||||
}),
|
||||
};
|
||||
this.rect.group.add(this.rect.fillRect);
|
||||
this.group.add(this.rect.group);
|
||||
}
|
||||
|
||||
|
@ -67,6 +67,7 @@ export class CanvasRegion {
|
||||
// Destroy any objects that are no longer in state
|
||||
for (const object of this.objects.values()) {
|
||||
if (!objectIds.includes(object.id)) {
|
||||
this.objects.delete(object.id);
|
||||
object.destroy();
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
@ -80,7 +81,7 @@ export class CanvasRegion {
|
||||
if (!brushLine) {
|
||||
brushLine = new KonvaBrushLine({ brushLine: obj });
|
||||
this.objects.set(brushLine.id, brushLine);
|
||||
this.konvaLayer.add(brushLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(brushLine.konvaLineGroup);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
|
||||
@ -95,7 +96,7 @@ export class CanvasRegion {
|
||||
if (!eraserLine) {
|
||||
eraserLine = new KonvaEraserLine({ eraserLine: obj });
|
||||
this.objects.set(eraserLine.id, eraserLine);
|
||||
this.konvaLayer.add(eraserLine.konvaLineGroup);
|
||||
this.konvaObjectGroup.add(eraserLine.konvaLineGroup);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
|
||||
@ -110,7 +111,7 @@ export class CanvasRegion {
|
||||
if (!rect) {
|
||||
rect = new KonvaRect({ rectShape: obj });
|
||||
this.objects.set(rect.id, rect);
|
||||
this.konvaLayer.add(rect.konvaRect);
|
||||
this.konvaObjectGroup.add(rect.konvaRect);
|
||||
groupNeedsCache = true;
|
||||
}
|
||||
}
|
||||
@ -128,48 +129,67 @@ export class CanvasRegion {
|
||||
return;
|
||||
}
|
||||
|
||||
const isSelected = selectedEntityIdentifier?.id === regionState.id;
|
||||
|
||||
/**
|
||||
* When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||
* shapes to render as a "raster" layer with all pixels drawn at the same color and opacity.
|
||||
*
|
||||
* Without this special handling, each shape is drawn individually with the given opacity, atop the other shapes. The
|
||||
* effect is like if you have a Photoshop Group consisting of many shapes, each of which has the given opacity.
|
||||
* Overlapping shapes will have their colors blended together, and the final color is the result of all the shapes.
|
||||
*
|
||||
* Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
||||
* a single raster image, and _then_ applied the 50% opacity.
|
||||
*/
|
||||
if (isSelected && selectedTool !== 'move') {
|
||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
if (this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.clearCache();
|
||||
}
|
||||
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
this.konvaObjectGroup.opacity(1);
|
||||
|
||||
this.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
|
||||
...(!regionState.bboxNeedsUpdate && regionState.bbox ? regionState.bbox : getLayerBboxFast(this.konvaLayer)),
|
||||
fill: rgbColor,
|
||||
opacity: maskOpacity,
|
||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
globalCompositeOperation: 'source-in',
|
||||
visible: true,
|
||||
// This rect must always be on top of all other shapes
|
||||
zIndex: this.objects.size + 1,
|
||||
});
|
||||
} else {
|
||||
// The compositing rect should only be shown when the layer is selected.
|
||||
this.compositingRect.visible(false);
|
||||
// Cache only if needed - or if we are on this code path and _don't_ have a cache
|
||||
if (groupNeedsCache || !this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.cache();
|
||||
}
|
||||
// Updating group opacity does not require re-caching
|
||||
this.konvaObjectGroup.opacity(maskOpacity);
|
||||
// We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
if (this.konvaObjectGroup.isCached()) {
|
||||
this.konvaObjectGroup.clearCache();
|
||||
}
|
||||
// The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
this.konvaObjectGroup.opacity(1);
|
||||
|
||||
this.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
|
||||
...(!regionState.bboxNeedsUpdate && regionState.bbox ? regionState.bbox : getLayerBboxFast(this.konvaLayer)),
|
||||
fill: rgbColor,
|
||||
opacity: maskOpacity,
|
||||
// Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
globalCompositeOperation: 'source-in',
|
||||
visible: true,
|
||||
// This rect must always be on top of all other shapes
|
||||
zIndex: this.objects.size + 1,
|
||||
});
|
||||
|
||||
// const isSelected = selectedEntityIdentifier?.id === regionState.id;
|
||||
|
||||
// /**
|
||||
// * When the group is selected, we use a rect of the selected preview color, composited over the shapes. This allows
|
||||
// * shapes to render as a "raster" layer with all pixels drawn at the same color and opacity.
|
||||
// *
|
||||
// * Without this special handling, each shape is drawn individually with the given opacity, atop the other shapes. The
|
||||
// * effect is like if you have a Photoshop Group consisting of many shapes, each of which has the given opacity.
|
||||
// * Overlapping shapes will have their colors blended together, and the final color is the result of all the shapes.
|
||||
// *
|
||||
// * Instead, with the special handling, the effect is as if you drew all the shapes at 100% opacity, flattened them to
|
||||
// * a single raster image, and _then_ applied the 50% opacity.
|
||||
// */
|
||||
// if (isSelected && selectedTool !== 'move') {
|
||||
// // We must clear the cache first so Konva will re-draw the group with the new compositing rect
|
||||
// if (this.konvaObjectGroup.isCached()) {
|
||||
// this.konvaObjectGroup.clearCache();
|
||||
// }
|
||||
// // The user is allowed to reduce mask opacity to 0, but we need the opacity for the compositing rect to work
|
||||
// this.konvaObjectGroup.opacity(1);
|
||||
|
||||
// this.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
|
||||
// ...(!regionState.bboxNeedsUpdate && regionState.bbox ? regionState.bbox : getLayerBboxFast(this.konvaLayer)),
|
||||
// fill: rgbColor,
|
||||
// opacity: maskOpacity,
|
||||
// // Draw this rect only where there are non-transparent pixels under it (e.g. the mask shapes)
|
||||
// globalCompositeOperation: 'source-in',
|
||||
// visible: true,
|
||||
// // This rect must always be on top of all other shapes
|
||||
// zIndex: this.objects.size + 1,
|
||||
// });
|
||||
// } else {
|
||||
// // The compositing rect should only be shown when the layer is selected.
|
||||
// this.compositingRect.visible(false);
|
||||
// // Cache only if needed - or if we are on this code path and _don't_ have a cache
|
||||
// if (groupNeedsCache || !this.konvaObjectGroup.isCached()) {
|
||||
// this.konvaObjectGroup.cache();
|
||||
// }
|
||||
// // Updating group opacity does not require re-caching
|
||||
// this.konvaObjectGroup.opacity(maskOpacity);
|
||||
// }
|
||||
|
||||
// const bboxRect =
|
||||
// regionMap.konvaLayer.findOne<Konva.Rect>(`.${LAYER_BBOX_NAME}`) ?? createBboxRect(rg, regionMap.konvaLayer);
|
||||
|
@ -18,6 +18,7 @@ import {
|
||||
imEraserLineAdded,
|
||||
imImageCacheChanged,
|
||||
imLinePointAdded,
|
||||
imRectAdded,
|
||||
imTranslated,
|
||||
layerBboxChanged,
|
||||
layerBrushLineAdded,
|
||||
@ -146,6 +147,8 @@ export const initializeRenderer = (
|
||||
dispatch(layerRectAdded(arg));
|
||||
} else if (entityType === 'regional_guidance') {
|
||||
dispatch(rgRectAdded(arg));
|
||||
} else if (entityType === 'inpaint_mask') {
|
||||
dispatch(imRectAdded(arg));
|
||||
}
|
||||
};
|
||||
const onBboxTransformed = (bbox: IRect) => {
|
||||
|
@ -20,7 +20,7 @@ import type { AspectRatioState } from 'features/parameters/components/ImageSize/
|
||||
import { atom } from 'nanostores';
|
||||
|
||||
import type { CanvasEntityIdentifier, CanvasV2State, StageAttrs } from './types';
|
||||
import { DEFAULT_RGBA_COLOR } from './types';
|
||||
import { RGBA_RED } from './types';
|
||||
|
||||
const initialState: CanvasV2State = {
|
||||
_version: 3,
|
||||
@ -35,7 +35,7 @@ const initialState: CanvasV2State = {
|
||||
type: 'inpaint_mask',
|
||||
bbox: null,
|
||||
bboxNeedsUpdate: false,
|
||||
fill: DEFAULT_RGBA_COLOR,
|
||||
fill: RGBA_RED,
|
||||
imageCache: null,
|
||||
isEnabled: true,
|
||||
objects: [],
|
||||
@ -46,7 +46,7 @@ const initialState: CanvasV2State = {
|
||||
selected: 'bbox',
|
||||
selectedBuffer: null,
|
||||
invertScroll: false,
|
||||
fill: DEFAULT_RGBA_COLOR,
|
||||
fill: RGBA_RED,
|
||||
brush: {
|
||||
width: 50,
|
||||
},
|
||||
|
@ -1,7 +1,7 @@
|
||||
import type { PayloadAction, SliceCaseReducers } from '@reduxjs/toolkit';
|
||||
import { getBrushLineId, getEraserLineId, getRectShapeId } from 'features/controlLayers/konva/naming';
|
||||
import type { CanvasV2State, InpaintMaskEntity } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageWithDims } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageWithDims,RGBA_RED } from 'features/controlLayers/store/types';
|
||||
import type { IRect } from 'konva/lib/types';
|
||||
import type { ImageDTO } from 'services/api/types';
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
@ -44,13 +44,13 @@ export const inpaintMaskReducers = {
|
||||
},
|
||||
imBrushLineAdded: {
|
||||
reducer: (state, action: PayloadAction<Omit<BrushLineAddedArg, 'id'> & { lineId: string }>) => {
|
||||
const { points, lineId, color, width, clip } = action.payload;
|
||||
const { points, lineId, width, clip } = action.payload;
|
||||
state.inpaintMask.objects.push({
|
||||
id: getBrushLineId(state.inpaintMask.id, lineId),
|
||||
type: 'brush_line',
|
||||
points,
|
||||
strokeWidth: width,
|
||||
color,
|
||||
color: RGBA_RED,
|
||||
clip,
|
||||
});
|
||||
state.inpaintMask.bboxNeedsUpdate = true;
|
||||
@ -89,7 +89,7 @@ export const inpaintMaskReducers = {
|
||||
},
|
||||
imRectAdded: {
|
||||
reducer: (state, action: PayloadAction<Omit<RectShapeAddedArg, 'id'> & { rectId: string }>) => {
|
||||
const { rect, rectId, color } = action.payload;
|
||||
const { rect, rectId } = action.payload;
|
||||
if (rect.height === 0 || rect.width === 0) {
|
||||
// Ignore zero-area rectangles
|
||||
return;
|
||||
@ -98,7 +98,7 @@ export const inpaintMaskReducers = {
|
||||
type: 'rect_shape',
|
||||
id: getRectShapeId(state.inpaintMask.id, rectId),
|
||||
...rect,
|
||||
color,
|
||||
color: RGBA_RED,
|
||||
});
|
||||
state.inpaintMask.bboxNeedsUpdate = true;
|
||||
state.inpaintMask.imageCache = null;
|
||||
|
@ -2,7 +2,7 @@ import type { PayloadAction, SliceCaseReducers } from '@reduxjs/toolkit';
|
||||
import { moveOneToEnd, moveOneToStart, moveToEnd, moveToStart } from 'common/util/arrayUtils';
|
||||
import { getBrushLineId, getEraserLineId, getRectShapeId } from 'features/controlLayers/konva/naming';
|
||||
import type { CanvasV2State, CLIPVisionModelV2, IPMethodV2 } from 'features/controlLayers/store/types';
|
||||
import { DEFAULT_RGBA_COLOR, imageDTOToImageObject, imageDTOToImageWithDims } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageObject, imageDTOToImageWithDims,RGBA_RED } from 'features/controlLayers/store/types';
|
||||
import { zModelIdentifierField } from 'features/nodes/types/common';
|
||||
import type { ParameterAutoNegative } from 'features/parameters/types/parameterSchemas';
|
||||
import type { IRect } from 'konva/lib/types';
|
||||
@ -319,7 +319,7 @@ export const regionsReducers = {
|
||||
type: 'brush_line',
|
||||
points,
|
||||
strokeWidth: width,
|
||||
color: DEFAULT_RGBA_COLOR,
|
||||
color: RGBA_RED,
|
||||
clip,
|
||||
});
|
||||
rg.bboxNeedsUpdate = true;
|
||||
@ -379,7 +379,7 @@ export const regionsReducers = {
|
||||
type: 'rect_shape',
|
||||
id: getRectShapeId(id, rectId),
|
||||
...rect,
|
||||
color: DEFAULT_RGBA_COLOR,
|
||||
color: RGBA_RED,
|
||||
});
|
||||
rg.bboxNeedsUpdate = true;
|
||||
rg.imageCache = null;
|
||||
|
@ -495,7 +495,7 @@ const zRgbaColor = zRgbColor.extend({
|
||||
a: z.number().min(0).max(1),
|
||||
});
|
||||
export type RgbaColor = z.infer<typeof zRgbaColor>;
|
||||
export const DEFAULT_RGBA_COLOR: RgbaColor = { r: 255, g: 255, b: 255, a: 1 };
|
||||
export const RGBA_RED: RgbaColor = { r: 255, g: 0, b: 0, a: 1 };
|
||||
|
||||
const zOpacity = z.number().gte(0).lte(1);
|
||||
|
||||
|
@ -18,7 +18,7 @@ export const addImageToImage = async (
|
||||
denoise.denoising_start = denoising_start;
|
||||
|
||||
const cropBbox = pick(bbox, ['x', 'y', 'width', 'height']);
|
||||
const initialImage = await manager.util.getImageSourceImage({
|
||||
const initialImage = await manager.getImageSourceImage({
|
||||
bbox: cropBbox,
|
||||
preview: true,
|
||||
});
|
||||
|
@ -22,11 +22,11 @@ export const addInpaint = async (
|
||||
denoise.denoising_start = denoising_start;
|
||||
|
||||
const cropBbox = pick(bbox, ['x', 'y', 'width', 'height']);
|
||||
const initialImage = await manager.util.getImageSourceImage({
|
||||
const initialImage = await manager.getImageSourceImage({
|
||||
bbox: cropBbox,
|
||||
preview: true,
|
||||
});
|
||||
const maskImage = await manager.util.getInpaintMaskImage({
|
||||
const maskImage = await manager.getInpaintMaskImage({
|
||||
bbox: cropBbox,
|
||||
preview: true,
|
||||
});
|
||||
|
@ -21,11 +21,11 @@ export const addOutpaint = async (
|
||||
vaePrecision: ParameterPrecision
|
||||
): Promise<Invocation<'canvas_paste_back'>> => {
|
||||
const cropBbox = pick(bbox, ['x', 'y', 'width', 'height']);
|
||||
const initialImage = await manager.util.getImageSourceImage({
|
||||
const initialImage = await manager.getImageSourceImage({
|
||||
bbox: cropBbox,
|
||||
preview: true,
|
||||
});
|
||||
const maskImage = await manager.util.getInpaintMaskImage({
|
||||
const maskImage = await manager.getInpaintMaskImage({
|
||||
bbox: cropBbox,
|
||||
preview: true,
|
||||
});
|
||||
|
@ -44,7 +44,7 @@ export const addRegions = async (
|
||||
|
||||
for (const region of validRegions) {
|
||||
// Upload the mask image, or get the cached image if it exists
|
||||
const { image_name } = await manager.util.getRegionMaskImage({ id: region.id, bbox, preview: true });
|
||||
const { image_name } = await manager.getRegionMaskImage({ id: region.id, bbox, preview: true });
|
||||
|
||||
// The main mask-to-tensor node
|
||||
const maskToTensor = g.addNode({
|
||||
|
@ -36,7 +36,7 @@ import { assert } from 'tsafe';
|
||||
import { addRegions } from './addRegions';
|
||||
|
||||
export const buildSD1Graph = async (state: RootState, manager: KonvaNodeManager): Promise<GraphType> => {
|
||||
const generationMode = manager.util.getGenerationMode();
|
||||
const generationMode = manager.getGenerationMode();
|
||||
|
||||
const { bbox, params } = state.canvasV2;
|
||||
|
||||
|
@ -34,7 +34,7 @@ import { assert } from 'tsafe';
|
||||
import { addRegions } from './addRegions';
|
||||
|
||||
export const buildSDXLGraph = async (state: RootState, manager: KonvaNodeManager): Promise<NonNullableGraph> => {
|
||||
const generationMode = manager.util.getGenerationMode();
|
||||
const generationMode = manager.getGenerationMode();
|
||||
|
||||
const { bbox, params } = state.canvasV2;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user