mirror of
https://github.com/comfyanonymous/ComfyUI.git
synced 2025-01-25 15:55:18 +00:00
Added UploadImage+Mask nodes
This commit is contained in:
parent
c70f0ac64b
commit
4a326a2548
1
.gitignore
vendored
1
.gitignore
vendored
@ -5,3 +5,4 @@ models/checkpoints
|
|||||||
models/vae
|
models/vae
|
||||||
models/embeddings
|
models/embeddings
|
||||||
models/loras
|
models/loras
|
||||||
|
uploads/
|
12
nodes.py
12
nodes.py
@ -807,6 +807,8 @@ class LoadImage:
|
|||||||
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "input")
|
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "input")
|
||||||
@classmethod
|
@classmethod
|
||||||
def INPUT_TYPES(s):
|
def INPUT_TYPES(s):
|
||||||
|
if not os.path.exists(s.input_dir):
|
||||||
|
os.makedirs(s.input_dir)
|
||||||
return {"required":
|
return {"required":
|
||||||
{"image": (sorted(os.listdir(s.input_dir)), )},
|
{"image": (sorted(os.listdir(s.input_dir)), )},
|
||||||
}
|
}
|
||||||
@ -830,7 +832,10 @@ class LoadImage:
|
|||||||
with open(image_path, 'rb') as f:
|
with open(image_path, 'rb') as f:
|
||||||
m.update(f.read())
|
m.update(f.read())
|
||||||
return m.digest().hex()
|
return m.digest().hex()
|
||||||
|
|
||||||
|
class UploadImage(LoadImage):
|
||||||
|
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "uploads")
|
||||||
|
|
||||||
class LoadImageMask:
|
class LoadImageMask:
|
||||||
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "input")
|
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "input")
|
||||||
@classmethod
|
@classmethod
|
||||||
@ -865,6 +870,9 @@ class LoadImageMask:
|
|||||||
with open(image_path, 'rb') as f:
|
with open(image_path, 'rb') as f:
|
||||||
m.update(f.read())
|
m.update(f.read())
|
||||||
return m.digest().hex()
|
return m.digest().hex()
|
||||||
|
|
||||||
|
class UploadImageMask(LoadImageMask):
|
||||||
|
input_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "uploads")
|
||||||
|
|
||||||
class ImageScale:
|
class ImageScale:
|
||||||
upscale_methods = ["nearest-exact", "bilinear", "area"]
|
upscale_methods = ["nearest-exact", "bilinear", "area"]
|
||||||
@ -917,7 +925,9 @@ NODE_CLASS_MAPPINGS = {
|
|||||||
"LatentUpscale": LatentUpscale,
|
"LatentUpscale": LatentUpscale,
|
||||||
"SaveImage": SaveImage,
|
"SaveImage": SaveImage,
|
||||||
"LoadImage": LoadImage,
|
"LoadImage": LoadImage,
|
||||||
|
"UploadImage": UploadImage,
|
||||||
"LoadImageMask": LoadImageMask,
|
"LoadImageMask": LoadImageMask,
|
||||||
|
"UploadImageMask": UploadImageMask,
|
||||||
"ImageScale": ImageScale,
|
"ImageScale": ImageScale,
|
||||||
"ImageInvert": ImageInvert,
|
"ImageInvert": ImageInvert,
|
||||||
"ConditioningCombine": ConditioningCombine,
|
"ConditioningCombine": ConditioningCombine,
|
||||||
|
32
server.py
32
server.py
@ -6,7 +6,6 @@ import execution
|
|||||||
import uuid
|
import uuid
|
||||||
import json
|
import json
|
||||||
import glob
|
import glob
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import aiohttp
|
import aiohttp
|
||||||
from aiohttp import web
|
from aiohttp import web
|
||||||
@ -27,7 +26,7 @@ class PromptServer():
|
|||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.messages = asyncio.Queue()
|
self.messages = asyncio.Queue()
|
||||||
self.number = 0
|
self.number = 0
|
||||||
self.app = web.Application()
|
self.app = web.Application(client_max_size=20971520)
|
||||||
self.sockets = dict()
|
self.sockets = dict()
|
||||||
self.web_root = os.path.join(os.path.dirname(
|
self.web_root = os.path.join(os.path.dirname(
|
||||||
os.path.realpath(__file__)), "web")
|
os.path.realpath(__file__)), "web")
|
||||||
@ -71,10 +70,37 @@ class PromptServer():
|
|||||||
files = glob.glob(os.path.join(self.web_root, 'extensions/**/*.js'), recursive=True)
|
files = glob.glob(os.path.join(self.web_root, 'extensions/**/*.js'), recursive=True)
|
||||||
return web.json_response(list(map(lambda f: "/" + os.path.relpath(f, self.web_root).replace("\\", "/"), files)))
|
return web.json_response(list(map(lambda f: "/" + os.path.relpath(f, self.web_root).replace("\\", "/"), files)))
|
||||||
|
|
||||||
|
@routes.post("/upload/image")
|
||||||
|
async def upload_image(request):
|
||||||
|
upload_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "uploads")
|
||||||
|
|
||||||
|
if not os.path.exists(upload_dir):
|
||||||
|
os.makedirs(upload_dir)
|
||||||
|
|
||||||
|
post = await request.post()
|
||||||
|
image = post.get("image")
|
||||||
|
|
||||||
|
if image and image.file:
|
||||||
|
filename = image.filename
|
||||||
|
if not filename:
|
||||||
|
return web.Response(status=400)
|
||||||
|
|
||||||
|
with open(os.path.join(upload_dir, filename), "wb") as f:
|
||||||
|
f.write(image.file.read())
|
||||||
|
|
||||||
|
return web.json_response({"name" : filename})
|
||||||
|
else:
|
||||||
|
return web.Response(status=400)
|
||||||
|
|
||||||
|
|
||||||
@routes.get("/view/{file}")
|
@routes.get("/view/{file}")
|
||||||
async def view_image(request):
|
async def view_image(request):
|
||||||
if "file" in request.match_info:
|
if "file" in request.match_info:
|
||||||
output_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "output")
|
type = request.rel_url.query.get("type", "output")
|
||||||
|
if type != "output" and type != "uploads":
|
||||||
|
return web.Response(status=400)
|
||||||
|
|
||||||
|
output_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), type)
|
||||||
file = request.match_info["file"]
|
file = request.match_info["file"]
|
||||||
file = os.path.splitext(os.path.basename(file))[0] + ".png"
|
file = os.path.splitext(os.path.basename(file))[0] + ".png"
|
||||||
file = os.path.join(output_dir, file)
|
file = os.path.join(output_dir, file)
|
||||||
|
10
web/extensions/core/uploadImage.js
Normal file
10
web/extensions/core/uploadImage.js
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import { app } from "/scripts/app.js";
|
||||||
|
|
||||||
|
app.registerExtension({
|
||||||
|
name: "Comfy.UploadImage",
|
||||||
|
async beforeRegisterNodeDef(nodeType, nodeData, app) {
|
||||||
|
if (nodeData.name === "UploadImage" || nodeData.name === "UploadImageMask") {
|
||||||
|
nodeData.input.required.upload = ["IMAGEUPLOAD"];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
@ -142,7 +142,7 @@ class ComfyApp {
|
|||||||
if (numImages === 1 && !imageIndex) {
|
if (numImages === 1 && !imageIndex) {
|
||||||
this.imageIndex = imageIndex = 0;
|
this.imageIndex = imageIndex = 0;
|
||||||
}
|
}
|
||||||
let shiftY = this.type === "SaveImage" ? 55 : 0;
|
let shiftY = this.type === "SaveImage" ? 55 : this.imageOffset || 0;
|
||||||
let dw = this.size[0];
|
let dw = this.size[0];
|
||||||
let dh = this.size[1];
|
let dh = this.size[1];
|
||||||
dh -= shiftY;
|
dh -= shiftY;
|
||||||
|
@ -126,4 +126,84 @@ export const ComfyWidgets = {
|
|||||||
return { widget: node.addWidget("text", inputName, defaultVal, () => {}, {}) };
|
return { widget: node.addWidget("text", inputName, defaultVal, () => {}, {}) };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
IMAGEUPLOAD(node, inputName, inputData, app) {
|
||||||
|
const imageWidget = node.widgets.find((w) => w.name === "image");
|
||||||
|
let uploadWidget;
|
||||||
|
|
||||||
|
function showImage(name) {
|
||||||
|
// Position the image somewhere sensible
|
||||||
|
if(!node.imageOffset) {
|
||||||
|
node.imageOffset = uploadWidget.last_y ? uploadWidget.last_y + 25 : 75;
|
||||||
|
}
|
||||||
|
|
||||||
|
const img = new Image();
|
||||||
|
img.onload = () => {
|
||||||
|
node.imgs = [img];
|
||||||
|
app.graph.setDirtyCanvas(true);
|
||||||
|
};
|
||||||
|
img.src = `/view/${name}?type=uploads`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add our own callback to the combo widget to render an image when it changes
|
||||||
|
const cb = node.callback;
|
||||||
|
imageWidget.callback = function () {
|
||||||
|
showImage(imageWidget.value);
|
||||||
|
if (cb) {
|
||||||
|
return cb.apply(this, arguments);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// On load if we have a value then render the image
|
||||||
|
// The value isnt set immediately so we need to wait a moment
|
||||||
|
// No change callbacks seem to be fired on initial setting of the value
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
if (imageWidget.value) {
|
||||||
|
showImage(imageWidget.value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const fileInput = document.createElement("input");
|
||||||
|
Object.assign(fileInput, {
|
||||||
|
type: "file",
|
||||||
|
accept: "image/jpeg,image/png",
|
||||||
|
style: "display: none",
|
||||||
|
onchange: async () => {
|
||||||
|
if (fileInput.files.length) {
|
||||||
|
try {
|
||||||
|
// Wrap file in formdata so it includes filename
|
||||||
|
const body = new FormData();
|
||||||
|
body.append("image", fileInput.files[0]);
|
||||||
|
const resp = await fetch("/upload/image", {
|
||||||
|
method: "POST",
|
||||||
|
body,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (resp.status === 200) {
|
||||||
|
const data = await resp.json();
|
||||||
|
showImage(data.name);
|
||||||
|
|
||||||
|
// Add the file as an option and update the widget value
|
||||||
|
if (!imageWidget.options.values.includes(data.name)) {
|
||||||
|
imageWidget.options.values.push(data.name);
|
||||||
|
}
|
||||||
|
imageWidget.value = data.name;
|
||||||
|
} else {
|
||||||
|
alert(resp.status + " - " + resp.statusText);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
alert(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
document.body.append(fileInput);
|
||||||
|
|
||||||
|
// Create the button widget for selecting the files
|
||||||
|
uploadWidget = node.addWidget("button", "choose file to upload", "image", () => {
|
||||||
|
fileInput.click();
|
||||||
|
});
|
||||||
|
uploadWidget.serialize = false;
|
||||||
|
|
||||||
|
return { widget: uploadWidget };
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user