|
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, |
|
}; |
|
} |
|
|
|
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; |
|
} |
|
|
|
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); |
|
}, |
|
}); |
|
|