mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
tidy(ui): abstract caching logic to module
This commit is contained in:
parent
68fad5cdcc
commit
7d2df399ed
@ -35,7 +35,7 @@ const ControlLayersSettingsPopover = () => {
|
|||||||
[dispatch]
|
[dispatch]
|
||||||
);
|
);
|
||||||
const clearCaches = useCallback(() => {
|
const clearCaches = useCallback(() => {
|
||||||
canvasManager?.clearCaches();
|
canvasManager?.cache.clearAll();
|
||||||
}, [canvasManager]);
|
}, [canvasManager]);
|
||||||
const calculateBboxes = useCallback(() => {
|
const calculateBboxes = useCallback(() => {
|
||||||
if (!canvasManager) {
|
if (!canvasManager) {
|
||||||
|
@ -0,0 +1,35 @@
|
|||||||
|
import type { SerializableObject } from 'common/types';
|
||||||
|
import type { CanvasManager } from 'features/controlLayers/konva/CanvasManager';
|
||||||
|
import { getPrefixedId } from 'features/controlLayers/konva/util';
|
||||||
|
import type { GenerationMode } from 'features/controlLayers/store/types';
|
||||||
|
import { LRUCache } from 'lru-cache';
|
||||||
|
import type { Logger } from 'roarr';
|
||||||
|
|
||||||
|
export class CanvasCacheModule {
|
||||||
|
id: string;
|
||||||
|
path: string[];
|
||||||
|
log: Logger;
|
||||||
|
manager: CanvasManager;
|
||||||
|
|
||||||
|
imageNameCache = new LRUCache<string, string>({ max: 100 });
|
||||||
|
canvasElementCache = new LRUCache<string, HTMLCanvasElement>({ max: 32 });
|
||||||
|
generationModeCache = new LRUCache<string, GenerationMode>({ max: 100 });
|
||||||
|
|
||||||
|
constructor(manager: CanvasManager) {
|
||||||
|
this.id = getPrefixedId('cache');
|
||||||
|
this.manager = manager;
|
||||||
|
this.path = this.manager.path.concat(this.id);
|
||||||
|
this.log = this.manager.buildLogger(this.getLoggingContext);
|
||||||
|
this.log.debug('Creating canvas cache');
|
||||||
|
}
|
||||||
|
|
||||||
|
clearAll = () => {
|
||||||
|
this.canvasElementCache.clear();
|
||||||
|
this.imageNameCache.clear();
|
||||||
|
this.generationModeCache.clear();
|
||||||
|
};
|
||||||
|
|
||||||
|
getLoggingContext = (): SerializableObject => {
|
||||||
|
return { ...this.manager.getLoggingContext(), path: this.path.join('.') };
|
||||||
|
};
|
||||||
|
}
|
@ -2,6 +2,7 @@ import type { AppSocket } from 'app/hooks/useSocketIO';
|
|||||||
import { logger } from 'app/logging/logger';
|
import { logger } from 'app/logging/logger';
|
||||||
import type { AppStore } from 'app/store/store';
|
import type { AppStore } from 'app/store/store';
|
||||||
import type { SerializableObject } from 'common/types';
|
import type { SerializableObject } from 'common/types';
|
||||||
|
import { CanvasCacheModule } from 'features/controlLayers/konva/CanvasCacheModule';
|
||||||
import { CanvasFilter } from 'features/controlLayers/konva/CanvasFilter';
|
import { CanvasFilter } from 'features/controlLayers/konva/CanvasFilter';
|
||||||
import { CanvasStageModule } from 'features/controlLayers/konva/CanvasStageModule';
|
import { CanvasStageModule } from 'features/controlLayers/konva/CanvasStageModule';
|
||||||
import { CanvasWorkerModule } from 'features/controlLayers/konva/CanvasWorkerModule.js';
|
import { CanvasWorkerModule } from 'features/controlLayers/konva/CanvasWorkerModule.js';
|
||||||
@ -14,7 +15,6 @@ import {
|
|||||||
} from 'features/controlLayers/konva/util';
|
} from 'features/controlLayers/konva/util';
|
||||||
import type { CanvasV2State, GenerationMode, Rect } from 'features/controlLayers/store/types';
|
import type { CanvasV2State, GenerationMode, Rect } from 'features/controlLayers/store/types';
|
||||||
import type Konva from 'konva';
|
import type Konva from 'konva';
|
||||||
import { LRUCache } from 'lru-cache';
|
|
||||||
import { atom } from 'nanostores';
|
import { atom } from 'nanostores';
|
||||||
import type { Logger } from 'roarr';
|
import type { Logger } from 'roarr';
|
||||||
import { getImageDTO, uploadImage } from 'services/api/endpoints/images';
|
import { getImageDTO, uploadImage } from 'services/api/endpoints/images';
|
||||||
@ -48,6 +48,7 @@ export class CanvasManager {
|
|||||||
filter: CanvasFilter;
|
filter: CanvasFilter;
|
||||||
stage: CanvasStageModule;
|
stage: CanvasStageModule;
|
||||||
worker: CanvasWorkerModule;
|
worker: CanvasWorkerModule;
|
||||||
|
cache: CanvasCacheModule;
|
||||||
|
|
||||||
log: Logger;
|
log: Logger;
|
||||||
socket: AppSocket;
|
socket: AppSocket;
|
||||||
@ -57,10 +58,6 @@ export class CanvasManager {
|
|||||||
isFirstRender: boolean = true;
|
isFirstRender: boolean = true;
|
||||||
_isDebugging: boolean = false;
|
_isDebugging: boolean = false;
|
||||||
|
|
||||||
imageNameCache = new LRUCache<string, string>({ max: 100 });
|
|
||||||
canvasCache = new LRUCache<string, HTMLCanvasElement>({ max: 32 });
|
|
||||||
generationModeCache = new LRUCache<string, GenerationMode>({ max: 100 });
|
|
||||||
|
|
||||||
constructor(stage: Konva.Stage, container: HTMLDivElement, store: AppStore, socket: AppSocket) {
|
constructor(stage: Konva.Stage, container: HTMLDivElement, store: AppStore, socket: AppSocket) {
|
||||||
this.id = getPrefixedId(this.type);
|
this.id = getPrefixedId(this.type);
|
||||||
this.path = [this.id];
|
this.path = [this.id];
|
||||||
@ -83,6 +80,7 @@ export class CanvasManager {
|
|||||||
|
|
||||||
this.stage = new CanvasStageModule(stage, container, this);
|
this.stage = new CanvasStageModule(stage, container, this);
|
||||||
this.worker = new CanvasWorkerModule(this);
|
this.worker = new CanvasWorkerModule(this);
|
||||||
|
this.cache = new CanvasCacheModule(this);
|
||||||
|
|
||||||
this.preview = new CanvasPreview(this);
|
this.preview = new CanvasPreview(this);
|
||||||
this.stage.addLayer(this.preview.getLayer());
|
this.stage.addLayer(this.preview.getLayer());
|
||||||
@ -404,12 +402,6 @@ export class CanvasManager {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
clearCaches = () => {
|
|
||||||
this.canvasCache.clear();
|
|
||||||
this.imageNameCache.clear();
|
|
||||||
this.generationModeCache.clear();
|
|
||||||
};
|
|
||||||
|
|
||||||
getCompositeRasterLayerEntityIds = (): string[] => {
|
getCompositeRasterLayerEntityIds = (): string[] => {
|
||||||
const ids = [];
|
const ids = [];
|
||||||
for (const adapter of this.rasterLayerAdapters.values()) {
|
for (const adapter of this.rasterLayerAdapters.values()) {
|
||||||
@ -432,7 +424,7 @@ export class CanvasManager {
|
|||||||
|
|
||||||
getCompositeRasterLayerCanvas = (rect: Rect): HTMLCanvasElement => {
|
getCompositeRasterLayerCanvas = (rect: Rect): HTMLCanvasElement => {
|
||||||
const hash = this.getCompositeRasterLayerHash({ rect });
|
const hash = this.getCompositeRasterLayerHash({ rect });
|
||||||
const cachedCanvas = this.canvasCache.get(hash);
|
const cachedCanvas = this.cache.canvasElementCache.get(hash);
|
||||||
|
|
||||||
if (cachedCanvas) {
|
if (cachedCanvas) {
|
||||||
this.log.trace({ rect }, 'Using cached composite inpaint mask canvas');
|
this.log.trace({ rect }, 'Using cached composite inpaint mask canvas');
|
||||||
@ -458,13 +450,13 @@ export class CanvasManager {
|
|||||||
const adapterCanvas = adapter.getCanvas(rect);
|
const adapterCanvas = adapter.getCanvas(rect);
|
||||||
ctx.drawImage(adapterCanvas, 0, 0);
|
ctx.drawImage(adapterCanvas, 0, 0);
|
||||||
}
|
}
|
||||||
this.canvasCache.set(hash, canvas);
|
this.cache.canvasElementCache.set(hash, canvas);
|
||||||
return canvas;
|
return canvas;
|
||||||
};
|
};
|
||||||
|
|
||||||
getCompositeInpaintMaskCanvas = (rect: Rect): HTMLCanvasElement => {
|
getCompositeInpaintMaskCanvas = (rect: Rect): HTMLCanvasElement => {
|
||||||
const hash = this.getCompositeInpaintMaskHash({ rect });
|
const hash = this.getCompositeInpaintMaskHash({ rect });
|
||||||
const cachedCanvas = this.canvasCache.get(hash);
|
const cachedCanvas = this.cache.canvasElementCache.get(hash);
|
||||||
|
|
||||||
if (cachedCanvas) {
|
if (cachedCanvas) {
|
||||||
this.log.trace({ rect }, 'Using cached composite inpaint mask canvas');
|
this.log.trace({ rect }, 'Using cached composite inpaint mask canvas');
|
||||||
@ -490,7 +482,7 @@ export class CanvasManager {
|
|||||||
const adapterCanvas = adapter.getCanvas(rect);
|
const adapterCanvas = adapter.getCanvas(rect);
|
||||||
ctx.drawImage(adapterCanvas, 0, 0);
|
ctx.drawImage(adapterCanvas, 0, 0);
|
||||||
}
|
}
|
||||||
this.canvasCache.set(hash, canvas);
|
this.cache.canvasElementCache.set(hash, canvas);
|
||||||
return canvas;
|
return canvas;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -528,7 +520,7 @@ export class CanvasManager {
|
|||||||
let imageDTO: ImageDTO | null = null;
|
let imageDTO: ImageDTO | null = null;
|
||||||
|
|
||||||
const hash = this.getCompositeRasterLayerHash({ rect });
|
const hash = this.getCompositeRasterLayerHash({ rect });
|
||||||
const cachedImageName = this.imageNameCache.get(hash);
|
const cachedImageName = this.cache.imageNameCache.get(hash);
|
||||||
|
|
||||||
if (cachedImageName) {
|
if (cachedImageName) {
|
||||||
imageDTO = await getImageDTO(cachedImageName);
|
imageDTO = await getImageDTO(cachedImageName);
|
||||||
@ -547,7 +539,7 @@ export class CanvasManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
imageDTO = await uploadImage(blob, 'composite-raster-layer.png', 'general', true);
|
imageDTO = await uploadImage(blob, 'composite-raster-layer.png', 'general', true);
|
||||||
this.imageNameCache.set(hash, imageDTO.image_name);
|
this.cache.imageNameCache.set(hash, imageDTO.image_name);
|
||||||
return imageDTO;
|
return imageDTO;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -555,7 +547,7 @@ export class CanvasManager {
|
|||||||
let imageDTO: ImageDTO | null = null;
|
let imageDTO: ImageDTO | null = null;
|
||||||
|
|
||||||
const hash = this.getCompositeInpaintMaskHash({ rect });
|
const hash = this.getCompositeInpaintMaskHash({ rect });
|
||||||
const cachedImageName = this.imageNameCache.get(hash);
|
const cachedImageName = this.cache.imageNameCache.get(hash);
|
||||||
|
|
||||||
if (cachedImageName) {
|
if (cachedImageName) {
|
||||||
imageDTO = await getImageDTO(cachedImageName);
|
imageDTO = await getImageDTO(cachedImageName);
|
||||||
@ -574,7 +566,7 @@ export class CanvasManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
imageDTO = await uploadImage(blob, 'composite-inpaint-mask.png', 'general', true);
|
imageDTO = await uploadImage(blob, 'composite-inpaint-mask.png', 'general', true);
|
||||||
this.imageNameCache.set(hash, imageDTO.image_name);
|
this.cache.imageNameCache.set(hash, imageDTO.image_name);
|
||||||
return imageDTO;
|
return imageDTO;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -584,7 +576,7 @@ export class CanvasManager {
|
|||||||
const compositeInpaintMaskHash = this.getCompositeInpaintMaskHash({ rect });
|
const compositeInpaintMaskHash = this.getCompositeInpaintMaskHash({ rect });
|
||||||
const compositeRasterLayerHash = this.getCompositeRasterLayerHash({ rect });
|
const compositeRasterLayerHash = this.getCompositeRasterLayerHash({ rect });
|
||||||
const hash = stableHash({ rect, compositeInpaintMaskHash, compositeRasterLayerHash });
|
const hash = stableHash({ rect, compositeInpaintMaskHash, compositeRasterLayerHash });
|
||||||
const cachedGenerationMode = this.generationModeCache.get(hash);
|
const cachedGenerationMode = this.cache.generationModeCache.get(hash);
|
||||||
|
|
||||||
if (cachedGenerationMode) {
|
if (cachedGenerationMode) {
|
||||||
this.log.trace({ rect, cachedGenerationMode }, 'Using cached generation mode');
|
this.log.trace({ rect, cachedGenerationMode }, 'Using cached generation mode');
|
||||||
@ -612,7 +604,7 @@ export class CanvasManager {
|
|||||||
generationMode = 'inpaint';
|
generationMode = 'inpaint';
|
||||||
}
|
}
|
||||||
|
|
||||||
this.generationModeCache.set(hash, generationMode);
|
this.cache.generationModeCache.set(hash, generationMode);
|
||||||
return generationMode;
|
return generationMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user