mirror of
https://github.com/Azornes/Comfyui-LayerForge.git
synced 2026-03-22 05:02:11 -03:00
Add localStorage state persistence and refactor image loading
Introduces methods to save and load canvas state to localStorage, enabling persistence across sessions. Refactors image loading for drag-and-drop, file input, and clipboard paste to use FileReader and data:URL, improving compatibility and reliability. Updates layer addition to consistently update selection and save state. Calls loadInitialState on widget creation to restore previous state if available.
This commit is contained in:
161
js/Canvas.js
161
js/Canvas.js
@@ -79,7 +79,115 @@ export class Canvas {
|
||||
this.redoStack = [];
|
||||
this.historyLimit = 100;
|
||||
|
||||
this.saveState();
|
||||
// this.saveState(); // Wywołanie przeniesione do loadInitialState
|
||||
}
|
||||
|
||||
getLocalStorageKey() {
|
||||
if (!this.node.id) {
|
||||
console.error("Node ID is not available for generating localStorage key.");
|
||||
return null;
|
||||
}
|
||||
return `canvas-state-${this.node.id}`;
|
||||
}
|
||||
|
||||
async loadStateFromLocalStorage() {
|
||||
console.log("Attempting to load state from localStorage for node:", this.node.id);
|
||||
const key = this.getLocalStorageKey();
|
||||
if (!key) return false;
|
||||
|
||||
try {
|
||||
const savedStateJSON = localStorage.getItem(key);
|
||||
if (!savedStateJSON) {
|
||||
console.log("No saved state found in localStorage for key:", key);
|
||||
return false;
|
||||
}
|
||||
console.log("Found saved state in localStorage:", savedStateJSON.substring(0, 200) + "...");
|
||||
|
||||
const savedState = JSON.parse(savedStateJSON);
|
||||
|
||||
this.width = savedState.width || 512;
|
||||
this.height = savedState.height || 512;
|
||||
this.viewport = savedState.viewport || { x: -(this.width / 4), y: -(this.height / 4), zoom: 0.8 };
|
||||
|
||||
this.updateCanvasSize(this.width, this.height, false);
|
||||
console.log(`Canvas resized to ${this.width}x${this.height} and viewport set.`);
|
||||
|
||||
const imagePromises = savedState.layers.map((layerData, index) => {
|
||||
return new Promise((resolve) => {
|
||||
if (layerData.imageSrc) {
|
||||
console.log(`Layer ${index}: Loading image from data:URL...`);
|
||||
const img = new Image();
|
||||
img.onload = () => {
|
||||
console.log(`Layer ${index}: Image loaded successfully.`);
|
||||
const newLayer = { ...layerData, image: img };
|
||||
delete newLayer.imageSrc;
|
||||
resolve(newLayer);
|
||||
};
|
||||
img.onerror = () => {
|
||||
console.error(`Layer ${index}: Failed to load image from src.`);
|
||||
resolve(null);
|
||||
};
|
||||
img.src = layerData.imageSrc;
|
||||
} else {
|
||||
console.log(`Layer ${index}: No imageSrc found, resolving layer data.`);
|
||||
resolve({ ...layerData });
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
const loadedLayers = await Promise.all(imagePromises);
|
||||
this.layers = loadedLayers.filter(l => l !== null);
|
||||
console.log(`Loaded ${this.layers.length} layers.`);
|
||||
|
||||
this.updateSelectionAfterHistory();
|
||||
this.render();
|
||||
console.log("Canvas state loaded successfully from localStorage for node", this.node.id);
|
||||
return true;
|
||||
} catch (e) {
|
||||
console.error("Error loading canvas state from localStorage:", e);
|
||||
localStorage.removeItem(key);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
saveStateToLocalStorage() {
|
||||
console.log("Attempting to save state to localStorage for node:", this.node.id);
|
||||
const key = this.getLocalStorageKey();
|
||||
if (!key) return;
|
||||
|
||||
try {
|
||||
const state = {
|
||||
layers: this.layers.map((layer, index) => {
|
||||
const newLayer = { ...layer };
|
||||
if (layer.image instanceof HTMLImageElement) {
|
||||
console.log(`Layer ${index}: Serializing image to data:URL.`);
|
||||
newLayer.imageSrc = layer.image.src;
|
||||
} else {
|
||||
console.log(`Layer ${index}: No HTMLImageElement found.`);
|
||||
}
|
||||
delete newLayer.image;
|
||||
return newLayer;
|
||||
}),
|
||||
viewport: this.viewport,
|
||||
width: this.width,
|
||||
height: this.height,
|
||||
};
|
||||
const stateJSON = JSON.stringify(state);
|
||||
localStorage.setItem(key, stateJSON);
|
||||
console.log("Canvas state saved to localStorage:", stateJSON.substring(0, 200) + "...");
|
||||
} catch (e) {
|
||||
console.error("Error saving canvas state to localStorage:", e);
|
||||
}
|
||||
}
|
||||
|
||||
async loadInitialState() {
|
||||
console.log("Loading initial state for node:", this.node.id);
|
||||
const loaded = await this.loadStateFromLocalStorage();
|
||||
if (!loaded) {
|
||||
console.log("No saved state found, initializing from node data.");
|
||||
await this.initNodeData();
|
||||
}
|
||||
this.saveState(); // Save initial state to undo stack
|
||||
}
|
||||
|
||||
cloneLayers(layers) {
|
||||
@@ -112,6 +220,7 @@ export class Canvas {
|
||||
}
|
||||
this.redoStack = [];
|
||||
this.updateHistoryButtons();
|
||||
this.saveStateToLocalStorage();
|
||||
}
|
||||
|
||||
undo() {
|
||||
@@ -309,25 +418,29 @@ export class Canvas {
|
||||
|
||||
if (imageType) {
|
||||
const blob = await item.getType(imageType);
|
||||
const img = new Image();
|
||||
img.onload = () => {
|
||||
const newLayer = {
|
||||
image: img,
|
||||
x: this.lastMousePosition.x - img.width / 2,
|
||||
y: this.lastMousePosition.y - img.height / 2,
|
||||
width: img.width,
|
||||
height: img.height,
|
||||
rotation: 0,
|
||||
zIndex: this.layers.length,
|
||||
blendMode: 'normal',
|
||||
opacity: 1
|
||||
const reader = new FileReader();
|
||||
reader.onload = (event) => {
|
||||
const img = new Image();
|
||||
img.onload = () => {
|
||||
const newLayer = {
|
||||
image: img,
|
||||
x: this.lastMousePosition.x - img.width / 2,
|
||||
y: this.lastMousePosition.y - img.height / 2,
|
||||
width: img.width,
|
||||
height: img.height,
|
||||
rotation: 0,
|
||||
zIndex: this.layers.length,
|
||||
blendMode: 'normal',
|
||||
opacity: 1
|
||||
};
|
||||
this.layers.push(newLayer);
|
||||
this.updateSelection([newLayer]);
|
||||
this.render();
|
||||
this.saveState();
|
||||
};
|
||||
this.layers.push(newLayer);
|
||||
this.updateSelection([newLayer]);
|
||||
this.render();
|
||||
URL.revokeObjectURL(img.src);
|
||||
img.src = event.target.result;
|
||||
};
|
||||
img.src = URL.createObjectURL(blob);
|
||||
reader.readAsDataURL(blob);
|
||||
imagePasted = true;
|
||||
break;
|
||||
}
|
||||
@@ -470,6 +583,7 @@ export class Canvas {
|
||||
this.viewport.y = worldCoords.y - (mouseBufferY / this.viewport.zoom);
|
||||
}
|
||||
this.render();
|
||||
this.saveState(true);
|
||||
}
|
||||
|
||||
handleKeyDown(e) {
|
||||
@@ -914,6 +1028,7 @@ export class Canvas {
|
||||
this.layers.push(layer);
|
||||
this.updateSelection([layer]);
|
||||
this.render();
|
||||
this.saveState();
|
||||
|
||||
console.log("Layer added successfully");
|
||||
} catch (error) {
|
||||
@@ -1010,8 +1125,10 @@ export class Canvas {
|
||||
this.render();
|
||||
}
|
||||
|
||||
updateCanvasSize(width, height) {
|
||||
this.saveState();
|
||||
updateCanvasSize(width, height, saveHistory = true) {
|
||||
if (saveHistory) {
|
||||
this.saveState();
|
||||
}
|
||||
this.width = width;
|
||||
this.height = height;
|
||||
|
||||
@@ -1019,6 +1136,10 @@ export class Canvas {
|
||||
this.canvas.height = height;
|
||||
|
||||
this.render();
|
||||
|
||||
if (saveHistory) {
|
||||
this.saveStateToLocalStorage();
|
||||
}
|
||||
}
|
||||
|
||||
render() {
|
||||
|
||||
@@ -271,41 +271,46 @@ async function createCanvasWidget(node, widget, app) {
|
||||
$el("button.painter-button.primary", {
|
||||
textContent: "Add Image",
|
||||
onclick: () => {
|
||||
console.log("Add Image button clicked.");
|
||||
const input = document.createElement('input');
|
||||
input.type = 'file';
|
||||
input.accept = 'image/*';
|
||||
input.multiple = true;
|
||||
input.onchange = async (e) => {
|
||||
for (const file of e.target.files) {
|
||||
console.log("File selected:", file.name);
|
||||
const reader = new FileReader();
|
||||
reader.onload = async (event) => {
|
||||
console.log("FileReader finished loading file as data:URL.");
|
||||
const img = new Image();
|
||||
img.onload = async () => {
|
||||
console.log("Image object loaded from data:URL.");
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
|
||||
const img = new Image();
|
||||
img.onload = async () => {
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length
|
||||
};
|
||||
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length
|
||||
canvas.layers.push(layer);
|
||||
canvas.updateSelection([layer]);
|
||||
canvas.render();
|
||||
canvas.saveState();
|
||||
console.log("New layer added and state saved.");
|
||||
await canvas.saveToServer(widget.value);
|
||||
app.graph.runStep();
|
||||
};
|
||||
|
||||
canvas.layers.push(layer);
|
||||
canvas.selectedLayer = layer;
|
||||
|
||||
canvas.render();
|
||||
|
||||
await canvas.saveToServer(widget.value);
|
||||
|
||||
app.graph.runStep();
|
||||
img.src = event.target.result;
|
||||
};
|
||||
img.src = URL.createObjectURL(file);
|
||||
reader.readAsDataURL(file);
|
||||
}
|
||||
};
|
||||
input.click();
|
||||
@@ -330,8 +335,10 @@ async function createCanvasWidget(node, widget, app) {
|
||||
$el("button.painter-button.primary", {
|
||||
textContent: "Paste Image",
|
||||
onclick: async () => {
|
||||
console.log("Paste Image button clicked.");
|
||||
try {
|
||||
if (!navigator.clipboard || !navigator.clipboard.read) {
|
||||
console.warn("Clipboard API not supported.");
|
||||
alert("Your browser does not support pasting from the clipboard.");
|
||||
return;
|
||||
}
|
||||
@@ -342,36 +349,45 @@ async function createCanvasWidget(node, widget, app) {
|
||||
const imageType = item.types.find(type => type.startsWith('image/'));
|
||||
|
||||
if (imageType) {
|
||||
console.log("Image found in clipboard.");
|
||||
const blob = await item.getType(imageType);
|
||||
const img = new Image();
|
||||
img.onload = () => {
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
const reader = new FileReader();
|
||||
reader.onload = (event) => {
|
||||
console.log("FileReader finished loading pasted blob as data:URL.");
|
||||
const img = new Image();
|
||||
img.onload = () => {
|
||||
console.log("Image object loaded from pasted data:URL.");
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length
|
||||
};
|
||||
|
||||
canvas.layers.push(layer);
|
||||
canvas.updateSelection([layer]);
|
||||
canvas.render();
|
||||
canvas.saveState();
|
||||
console.log("Pasted layer added and state saved.");
|
||||
};
|
||||
|
||||
canvas.layers.push(layer);
|
||||
canvas.updateSelection([layer]);
|
||||
canvas.render();
|
||||
URL.revokeObjectURL(img.src);
|
||||
img.src = event.target.result;
|
||||
};
|
||||
img.src = URL.createObjectURL(blob);
|
||||
reader.readAsDataURL(blob);
|
||||
imageFound = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!imageFound) {
|
||||
console.warn("No image found in clipboard.");
|
||||
alert("No image found in the clipboard.");
|
||||
}
|
||||
|
||||
@@ -695,36 +711,45 @@ async function createCanvasWidget(node, widget, app) {
|
||||
}
|
||||
}, [controlPanel, canvasContainer]);
|
||||
const handleFileLoad = async (file) => {
|
||||
console.log("File dropped:", file.name);
|
||||
if (!file.type.startsWith('image/')) {
|
||||
console.log("Dropped file is not an image.");
|
||||
return;
|
||||
}
|
||||
|
||||
const img = new Image();
|
||||
img.onload = async () => {
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
const reader = new FileReader();
|
||||
reader.onload = async (event) => {
|
||||
console.log("FileReader finished loading dropped file as data:URL.");
|
||||
const img = new Image();
|
||||
img.onload = async () => {
|
||||
console.log("Image object loaded from dropped data:URL.");
|
||||
const scale = Math.min(
|
||||
canvas.width / img.width,
|
||||
canvas.height / img.height
|
||||
);
|
||||
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length,
|
||||
blendMode: 'normal',
|
||||
opacity: 1
|
||||
const layer = {
|
||||
image: img,
|
||||
x: (canvas.width - img.width * scale) / 2,
|
||||
y: (canvas.height - img.height * scale) / 2,
|
||||
width: img.width * scale,
|
||||
height: img.height * scale,
|
||||
rotation: 0,
|
||||
zIndex: canvas.layers.length,
|
||||
blendMode: 'normal',
|
||||
opacity: 1
|
||||
};
|
||||
|
||||
canvas.layers.push(layer);
|
||||
canvas.updateSelection([layer]);
|
||||
canvas.render();
|
||||
canvas.saveState();
|
||||
console.log("Dropped layer added and state saved.");
|
||||
await updateOutput();
|
||||
};
|
||||
|
||||
canvas.layers.push(layer);
|
||||
canvas.selectedLayer = layer;
|
||||
canvas.render();
|
||||
await updateOutput();
|
||||
URL.revokeObjectURL(img.src);
|
||||
img.src = event.target.result;
|
||||
};
|
||||
img.src = URL.createObjectURL(file);
|
||||
reader.readAsDataURL(file);
|
||||
};
|
||||
|
||||
mainContainer.addEventListener('dragover', (e) => {
|
||||
@@ -775,6 +800,10 @@ async function createCanvasWidget(node, widget, app) {
|
||||
|
||||
node.canvasWidget = canvas;
|
||||
|
||||
setTimeout(() => {
|
||||
canvas.loadInitialState();
|
||||
}, 100);
|
||||
|
||||
return {
|
||||
canvas: canvas,
|
||||
panel: controlPanel
|
||||
@@ -1044,9 +1073,11 @@ app.registerExtension({
|
||||
if (nodeType.comfyClass === "CanvasNode") {
|
||||
const onNodeCreated = nodeType.prototype.onNodeCreated;
|
||||
nodeType.prototype.onNodeCreated = async function () {
|
||||
console.log("CanvasNode created, ID:", this.id);
|
||||
const r = onNodeCreated?.apply(this, arguments);
|
||||
|
||||
const widget = this.widgets.find(w => w.name === "canvas_image");
|
||||
console.log("Found canvas_image widget:", widget);
|
||||
await createCanvasWidget(this, widget, app);
|
||||
|
||||
return r;
|
||||
|
||||
Reference in New Issue
Block a user