|
import { app } from "../../../scripts/app.js";
|
|
import { importA1111 } from "../../../scripts/pnginfo.js";
|
|
import { ComfyWidgets } from "../../../scripts/widgets.js";
|
|
|
|
let getDrawTextConfig = null;
|
|
let fileInput;
|
|
|
|
class WorkflowImage {
|
|
static accept = "";
|
|
|
|
getBounds() {
|
|
|
|
const bounds = app.graph._nodes.reduce(
|
|
(p, n) => {
|
|
if (n.pos[0] < p[0]) p[0] = n.pos[0];
|
|
if (n.pos[1] < p[1]) p[1] = n.pos[1];
|
|
const bounds = n.getBounding();
|
|
const r = n.pos[0] + bounds[2];
|
|
const b = n.pos[1] + bounds[3];
|
|
if (r > p[2]) p[2] = r;
|
|
if (b > p[3]) p[3] = b;
|
|
return p;
|
|
},
|
|
[99999, 99999, -99999, -99999]
|
|
);
|
|
|
|
bounds[0] -= 100;
|
|
bounds[1] -= 100;
|
|
bounds[2] += 100;
|
|
bounds[3] += 100;
|
|
return bounds;
|
|
}
|
|
|
|
saveState() {
|
|
this.state = {
|
|
scale: app.canvas.ds.scale,
|
|
width: app.canvas.canvas.width,
|
|
height: app.canvas.canvas.height,
|
|
offset: app.canvas.ds.offset,
|
|
transform: app.canvas.canvas.getContext('2d').getTransform(),
|
|
};
|
|
}
|
|
|
|
restoreState() {
|
|
app.canvas.ds.scale = this.state.scale;
|
|
app.canvas.canvas.width = this.state.width;
|
|
app.canvas.canvas.height = this.state.height;
|
|
app.canvas.ds.offset = this.state.offset;
|
|
app.canvas.canvas.getContext('2d').setTransform(this.state.transform);
|
|
}
|
|
|
|
updateView(bounds) {
|
|
app.canvas.ds.scale = 1;
|
|
app.canvas.canvas.width = bounds[2] - bounds[0];
|
|
app.canvas.canvas.height = bounds[3] - bounds[1];
|
|
app.canvas.ds.offset = [-bounds[0], -bounds[1]];
|
|
}
|
|
|
|
getDrawTextConfig(_, widget) {
|
|
return {
|
|
x: 10,
|
|
y: widget.last_y + 10,
|
|
resetTransform: false,
|
|
};
|
|
}
|
|
|
|
async export(includeWorkflow) {
|
|
|
|
this.saveState();
|
|
|
|
this.updateView(this.getBounds());
|
|
|
|
|
|
getDrawTextConfig = this.getDrawTextConfig;
|
|
app.canvas.draw(true, true);
|
|
getDrawTextConfig = null;
|
|
|
|
|
|
const blob = await this.getBlob(includeWorkflow ? JSON.stringify(app.graph.serialize()) : undefined);
|
|
|
|
|
|
this.restoreState();
|
|
app.canvas.draw(true, true);
|
|
|
|
|
|
this.download(blob);
|
|
}
|
|
|
|
download(blob) {
|
|
const url = URL.createObjectURL(blob);
|
|
const a = document.createElement("a");
|
|
Object.assign(a, {
|
|
href: url,
|
|
download: "workflow." + this.extension,
|
|
style: "display: none",
|
|
});
|
|
document.body.append(a);
|
|
a.click();
|
|
setTimeout(function () {
|
|
a.remove();
|
|
window.URL.revokeObjectURL(url);
|
|
}, 0);
|
|
}
|
|
|
|
static import() {
|
|
if (!fileInput) {
|
|
fileInput = document.createElement("input");
|
|
Object.assign(fileInput, {
|
|
type: "file",
|
|
style: "display: none",
|
|
onchange: () => {
|
|
app.handleFile(fileInput.files[0]);
|
|
},
|
|
});
|
|
document.body.append(fileInput);
|
|
}
|
|
fileInput.accept = WorkflowImage.accept;
|
|
fileInput.click();
|
|
}
|
|
}
|
|
|
|
class PngWorkflowImage extends WorkflowImage {
|
|
static accept = ".png,image/png";
|
|
extension = "png";
|
|
|
|
n2b(n) {
|
|
return new Uint8Array([(n >> 24) & 0xff, (n >> 16) & 0xff, (n >> 8) & 0xff, n & 0xff]);
|
|
}
|
|
|
|
joinArrayBuffer(...bufs) {
|
|
const result = new Uint8Array(bufs.reduce((totalSize, buf) => totalSize + buf.byteLength, 0));
|
|
bufs.reduce((offset, buf) => {
|
|
result.set(buf, offset);
|
|
return offset + buf.byteLength;
|
|
}, 0);
|
|
return result;
|
|
}
|
|
|
|
crc32(data) {
|
|
const crcTable =
|
|
PngWorkflowImage.crcTable ||
|
|
(PngWorkflowImage.crcTable = (() => {
|
|
let c;
|
|
const crcTable = [];
|
|
for (let n = 0; n < 256; n++) {
|
|
c = n;
|
|
for (let k = 0; k < 8; k++) {
|
|
c = c & 1 ? 0xedb88320 ^ (c >>> 1) : c >>> 1;
|
|
}
|
|
crcTable[n] = c;
|
|
}
|
|
return crcTable;
|
|
})());
|
|
let crc = 0 ^ -1;
|
|
for (let i = 0; i < data.byteLength; i++) {
|
|
crc = (crc >>> 8) ^ crcTable[(crc ^ data[i]) & 0xff];
|
|
}
|
|
return (crc ^ -1) >>> 0;
|
|
}
|
|
|
|
async getBlob(workflow) {
|
|
return new Promise((r) => {
|
|
app.canvasEl.toBlob(async (blob) => {
|
|
if (workflow) {
|
|
|
|
const buffer = await blob.arrayBuffer();
|
|
const typedArr = new Uint8Array(buffer);
|
|
const view = new DataView(buffer);
|
|
|
|
const data = new TextEncoder().encode(`tEXtworkflow\0${workflow}`);
|
|
const chunk = this.joinArrayBuffer(this.n2b(data.byteLength - 4), data, this.n2b(this.crc32(data)));
|
|
|
|
const sz = view.getUint32(8) + 20;
|
|
const result = this.joinArrayBuffer(typedArr.subarray(0, sz), chunk, typedArr.subarray(sz));
|
|
|
|
blob = new Blob([result], { type: "image/png" });
|
|
}
|
|
|
|
r(blob);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
class DataReader {
|
|
|
|
view;
|
|
|
|
littleEndian;
|
|
offset = 0;
|
|
|
|
|
|
|
|
|
|
constructor(view) {
|
|
this.view = view;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
read(size, signed = false, littleEndian = undefined) {
|
|
const v = this.peek(size, signed, littleEndian);
|
|
this.offset += size;
|
|
return v;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
peek(size, signed = false, littleEndian = undefined) {
|
|
this.view.getBigInt64;
|
|
let m = "";
|
|
if (size === 8) m += "Big";
|
|
m += signed ? "Int" : "Uint";
|
|
m += size * 8;
|
|
m = "get" + m;
|
|
if (!this.view[m]) {
|
|
throw new Error("Method not found: " + m);
|
|
}
|
|
|
|
return this.view[m](this.offset, littleEndian == null ? this.littleEndian : littleEndian);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
seek(pos, relative = true) {
|
|
if (relative) {
|
|
this.offset += pos;
|
|
} else {
|
|
this.offset = pos;
|
|
}
|
|
}
|
|
}
|
|
|
|
class Tiff {
|
|
|
|
#reader;
|
|
#start;
|
|
|
|
readExif(reader) {
|
|
const TIFF_MARKER = 0x2a;
|
|
const EXIF_IFD = 0x8769;
|
|
|
|
this.#reader = reader;
|
|
this.#start = this.#reader.offset;
|
|
this.#readEndianness();
|
|
|
|
if (!this.#reader.read(2) === TIFF_MARKER) {
|
|
throw new Error("Invalid TIFF: Marker not found.");
|
|
}
|
|
|
|
const dirOffset = this.#reader.read(4);
|
|
this.#reader.seek(this.#start + dirOffset, false);
|
|
|
|
for (const t of this.#readTags()) {
|
|
if (t.id === EXIF_IFD) {
|
|
return this.#readExifTag(t);
|
|
}
|
|
}
|
|
throw new Error("No EXIF: TIFF Exif IFD tag not found");
|
|
}
|
|
|
|
#readUserComment(tag) {
|
|
this.#reader.seek(this.#start + tag.offset, false);
|
|
const encoding = this.#reader.read(8);
|
|
if (encoding !== 0x45444f43494e55n) {
|
|
throw new Error("Unable to read non-Unicode data");
|
|
}
|
|
const decoder = new TextDecoder("utf-16be");
|
|
return decoder.decode(new DataView(this.#reader.view.buffer, this.#reader.offset, tag.count - 8));
|
|
}
|
|
|
|
#readExifTag(exifTag) {
|
|
const EXIF_USER_COMMENT = 0x9286;
|
|
|
|
this.#reader.seek(this.#start + exifTag.offset, false);
|
|
for (const t of this.#readTags()) {
|
|
if (t.id === EXIF_USER_COMMENT) {
|
|
return this.#readUserComment(t);
|
|
}
|
|
}
|
|
throw new Error("No embedded data: UserComment Exif tag not found");
|
|
}
|
|
|
|
*#readTags() {
|
|
const count = this.#reader.read(2);
|
|
for (let i = 0; i < count; i++) {
|
|
yield {
|
|
id: this.#reader.read(2),
|
|
type: this.#reader.read(2),
|
|
count: this.#reader.read(4),
|
|
offset: this.#reader.read(4),
|
|
};
|
|
}
|
|
}
|
|
|
|
#readEndianness() {
|
|
const II = 0x4949;
|
|
const MM = 0x4d4d;
|
|
const endianness = this.#reader.read(2);
|
|
if (endianness === II) {
|
|
this.#reader.littleEndian = true;
|
|
} else if (endianness === MM) {
|
|
this.#reader.littleEndian = false;
|
|
} else {
|
|
throw new Error("Invalid JPEG: Endianness marker not found.");
|
|
}
|
|
}
|
|
}
|
|
|
|
class Jpeg {
|
|
|
|
#reader;
|
|
|
|
|
|
|
|
|
|
readExif(buffer) {
|
|
const JPEG_MARKER = 0xffd8;
|
|
const EXIF_SIG = 0x45786966;
|
|
|
|
this.#reader = new DataReader(new DataView(buffer));
|
|
if (!this.#reader.read(2) === JPEG_MARKER) {
|
|
throw new Error("Invalid JPEG: SOI not found.");
|
|
}
|
|
|
|
const app0 = this.#readAppMarkerId();
|
|
if (app0 !== 0) {
|
|
throw new Error(`Invalid JPEG: APP0 not found [found: ${app0}].`);
|
|
}
|
|
|
|
this.#consumeAppSegment();
|
|
const app1 = this.#readAppMarkerId();
|
|
if (app1 !== 1) {
|
|
throw new Error(`No EXIF: APP1 not found [found: ${app0}].`);
|
|
}
|
|
|
|
|
|
this.#reader.seek(2);
|
|
|
|
if (this.#reader.read(4) !== EXIF_SIG) {
|
|
throw new Error(`No EXIF: Invalid EXIF header signature.`);
|
|
}
|
|
if (this.#reader.read(2) !== 0) {
|
|
throw new Error(`No EXIF: Invalid EXIF header.`);
|
|
}
|
|
|
|
return new Tiff().readExif(this.#reader);
|
|
}
|
|
|
|
#readAppMarkerId() {
|
|
const APP0_MARKER = 0xffe0;
|
|
return this.#reader.read(2) - APP0_MARKER;
|
|
}
|
|
|
|
#consumeAppSegment() {
|
|
this.#reader.seek(this.#reader.read(2) - 2);
|
|
}
|
|
}
|
|
|
|
class SvgWorkflowImage extends WorkflowImage {
|
|
static accept = ".svg,image/svg+xml";
|
|
extension = "svg";
|
|
|
|
static init() {
|
|
|
|
const handleFile = app.handleFile;
|
|
app.handleFile = async function (file) {
|
|
if (file && (file.type === "image/svg+xml" || file.name?.endsWith(".svg"))) {
|
|
const reader = new FileReader();
|
|
reader.onload = () => {
|
|
|
|
const descEnd = reader.result.lastIndexOf("</desc>");
|
|
if (descEnd !== -1) {
|
|
const descStart = reader.result.lastIndexOf("<desc>", descEnd);
|
|
if (descStart !== -1) {
|
|
const json = reader.result.substring(descStart + 6, descEnd);
|
|
this.loadGraphData(JSON.parse(SvgWorkflowImage.unescapeXml(json)));
|
|
}
|
|
}
|
|
};
|
|
reader.readAsText(file);
|
|
return;
|
|
} else if (file && (file.type === "image/jpeg" || file.name?.endsWith(".jpg") || file.name?.endsWith(".jpeg"))) {
|
|
if (
|
|
await new Promise((resolve) => {
|
|
try {
|
|
|
|
const reader = new FileReader();
|
|
reader.onload = async () => {
|
|
try {
|
|
const value = new Jpeg().readExif(reader.result);
|
|
importA1111(app.graph, value);
|
|
resolve(true);
|
|
} catch (error) {
|
|
resolve(false);
|
|
}
|
|
};
|
|
reader.onerror = () => resolve(false);
|
|
reader.readAsArrayBuffer(file);
|
|
} catch (error) {
|
|
resolve(false);
|
|
}
|
|
})
|
|
) {
|
|
return;
|
|
}
|
|
}
|
|
return handleFile.apply(this, arguments);
|
|
};
|
|
}
|
|
|
|
static escapeXml(unsafe) {
|
|
return unsafe.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">");
|
|
}
|
|
|
|
static unescapeXml(safe) {
|
|
return safe.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">");
|
|
}
|
|
|
|
getDrawTextConfig(_, widget) {
|
|
return {
|
|
x: parseInt(widget.inputEl.style.left),
|
|
y: parseInt(widget.inputEl.style.top),
|
|
resetTransform: true,
|
|
};
|
|
}
|
|
|
|
saveState() {
|
|
super.saveState();
|
|
this.state.ctx = app.canvas.ctx;
|
|
}
|
|
|
|
restoreState() {
|
|
super.restoreState();
|
|
app.canvas.ctx = this.state.ctx;
|
|
}
|
|
|
|
updateView(bounds) {
|
|
super.updateView(bounds);
|
|
this.createSvgCtx(bounds);
|
|
}
|
|
|
|
createSvgCtx(bounds) {
|
|
const ctx = this.state.ctx;
|
|
const svgCtx = (this.svgCtx = new C2S(bounds[2] - bounds[0], bounds[3] - bounds[1]));
|
|
svgCtx.canvas.getBoundingClientRect = function () {
|
|
return { width: svgCtx.width, height: svgCtx.height };
|
|
};
|
|
|
|
|
|
const drawImage = svgCtx.drawImage;
|
|
svgCtx.drawImage = function (...args) {
|
|
const image = args[0];
|
|
|
|
|
|
if (image.nodeName === "IMG" && !image.src.startsWith("data:image/")) {
|
|
const canvas = document.createElement("canvas");
|
|
canvas.width = image.width;
|
|
canvas.height = image.height;
|
|
const imgCtx = canvas.getContext("2d");
|
|
imgCtx.drawImage(image, 0, 0);
|
|
args[0] = canvas;
|
|
}
|
|
|
|
return drawImage.apply(this, args);
|
|
};
|
|
|
|
|
|
svgCtx.getTransform = function () {
|
|
return ctx.getTransform();
|
|
};
|
|
svgCtx.resetTransform = function () {
|
|
return ctx.resetTransform();
|
|
};
|
|
svgCtx.roundRect = svgCtx.rect;
|
|
app.canvas.ctx = svgCtx;
|
|
}
|
|
|
|
getBlob(workflow) {
|
|
let svg = this.svgCtx
|
|
.getSerializedSvg(true)
|
|
.replace("<svg ", `<svg style="background: ${app.canvas.clear_background_color}" `);
|
|
|
|
if (workflow) {
|
|
svg = svg.replace("</svg>", `<desc>${SvgWorkflowImage.escapeXml(workflow)}</desc></svg>`);
|
|
}
|
|
|
|
return new Blob([svg], { type: "image/svg+xml" });
|
|
}
|
|
}
|
|
|
|
app.registerExtension({
|
|
name: "pysssss.WorkflowImage",
|
|
init() {
|
|
|
|
function wrapText(context, text, x, y, maxWidth, lineHeight) {
|
|
var words = text.split(" "),
|
|
line = "",
|
|
i,
|
|
test,
|
|
metrics;
|
|
|
|
for (i = 0; i < words.length; i++) {
|
|
test = words[i];
|
|
metrics = context.measureText(test);
|
|
while (metrics.width > maxWidth) {
|
|
|
|
test = test.substring(0, test.length - 1);
|
|
metrics = context.measureText(test);
|
|
}
|
|
if (words[i] != test) {
|
|
words.splice(i + 1, 0, words[i].substr(test.length));
|
|
words[i] = test;
|
|
}
|
|
|
|
test = line + words[i] + " ";
|
|
metrics = context.measureText(test);
|
|
|
|
if (metrics.width > maxWidth && i > 0) {
|
|
context.fillText(line, x, y);
|
|
line = words[i] + " ";
|
|
y += lineHeight;
|
|
} else {
|
|
line = test;
|
|
}
|
|
}
|
|
|
|
context.fillText(line, x, y);
|
|
}
|
|
|
|
const stringWidget = ComfyWidgets.STRING;
|
|
|
|
ComfyWidgets.STRING = function () {
|
|
const w = stringWidget.apply(this, arguments);
|
|
if (w.widget && w.widget.type === "customtext") {
|
|
const draw = w.widget.draw;
|
|
w.widget.draw = function (ctx) {
|
|
draw.apply(this, arguments);
|
|
if (this.inputEl.hidden) return;
|
|
|
|
if (getDrawTextConfig) {
|
|
const config = getDrawTextConfig(ctx, this);
|
|
const t = ctx.getTransform();
|
|
ctx.save();
|
|
if (config.resetTransform) {
|
|
ctx.resetTransform();
|
|
}
|
|
|
|
const style = document.defaultView.getComputedStyle(this.inputEl, null);
|
|
const x = config.x;
|
|
const y = config.y;
|
|
const w = parseInt(this.inputEl.style.width);
|
|
const h = parseInt(this.inputEl.style.height);
|
|
ctx.fillStyle = style.getPropertyValue("background-color");
|
|
ctx.fillRect(x, y, w, h);
|
|
|
|
ctx.fillStyle = style.getPropertyValue("color");
|
|
ctx.font = style.getPropertyValue("font");
|
|
|
|
const line = t.d * 12;
|
|
const split = this.inputEl.value.split("\n");
|
|
let start = y;
|
|
for (const l of split) {
|
|
start += line;
|
|
wrapText(ctx, l, x + 4, start, w, line);
|
|
}
|
|
|
|
ctx.restore();
|
|
}
|
|
};
|
|
}
|
|
return w;
|
|
};
|
|
},
|
|
setup() {
|
|
const script = document.createElement("script");
|
|
script.onload = function () {
|
|
const formats = [SvgWorkflowImage, PngWorkflowImage];
|
|
for (const f of formats) {
|
|
f.init?.call();
|
|
WorkflowImage.accept += (WorkflowImage.accept ? "," : "") + f.accept;
|
|
}
|
|
|
|
|
|
const orig = LGraphCanvas.prototype.getCanvasMenuOptions;
|
|
LGraphCanvas.prototype.getCanvasMenuOptions = function () {
|
|
const options = orig.apply(this, arguments);
|
|
|
|
options.push(null, {
|
|
content: "Workflow Image",
|
|
submenu: {
|
|
options: [
|
|
{
|
|
content: "Import",
|
|
callback: () => {
|
|
WorkflowImage.import();
|
|
},
|
|
},
|
|
{
|
|
content: "Export",
|
|
submenu: {
|
|
options: formats.flatMap((f) => [
|
|
{
|
|
content: f.name.replace("WorkflowImage", "").toLocaleLowerCase(),
|
|
callback: () => {
|
|
new f().export(true);
|
|
},
|
|
},
|
|
{
|
|
content: f.name.replace("WorkflowImage", "").toLocaleLowerCase() + " (no embedded workflow)",
|
|
callback: () => {
|
|
new f().export();
|
|
},
|
|
},
|
|
]),
|
|
},
|
|
},
|
|
],
|
|
},
|
|
});
|
|
return options;
|
|
};
|
|
};
|
|
|
|
script.src = new URL(`assets/canvas2svg.js`, import.meta.url);
|
|
document.body.append(script);
|
|
},
|
|
});
|
|
|