Spaces:
Sleeping
Sleeping
import { api } from "../../scripts/api.js"; | |
import { app } from "../../scripts/app.js"; | |
import { $el, ComfyDialog } from "../../scripts/ui.js"; | |
import { CopusShareDialog } from "./comfyui-share-copus.js"; | |
import { OpenArtShareDialog } from "./comfyui-share-openart.js"; | |
import { YouMLShareDialog } from "./comfyui-share-youml.js"; | |
export const SUPPORTED_OUTPUT_NODE_TYPES = [ | |
"PreviewImage", | |
"SaveImage", | |
"VHS_VideoCombine", | |
"ADE_AnimateDiffCombine", | |
"SaveAnimatedWEBP", | |
"CR Image Output" | |
] | |
var docStyle = document.createElement('style'); | |
docStyle.innerHTML = ` | |
.cm-menu-container { | |
column-gap: 20px; | |
display: flex; | |
flex-wrap: wrap; | |
justify-content: center; | |
} | |
.cm-menu-column { | |
display: flex; | |
flex-direction: column; | |
} | |
.cm-title { | |
padding: 10px 10px 0 10p; | |
background-color: black; | |
text-align: center; | |
height: 45px; | |
} | |
`; | |
document.head.appendChild(docStyle); | |
export function getPotentialOutputsAndOutputNodes(nodes) { | |
const potential_outputs = []; | |
const potential_output_nodes = []; | |
// iterate over the array of nodes to find the ones that are marked as SaveImage | |
// TODO: Add support for AnimateDiffCombine, etc. nodes that save videos/gifs, etc. | |
for (let i = 0; i < nodes.length; i++) { | |
const node = nodes[i]; | |
if (!SUPPORTED_OUTPUT_NODE_TYPES.includes(node.type)) { | |
continue; | |
} | |
if (node.type === "SaveImage" || node.type === "CR Image Output") { | |
// check if node has an 'images' array property | |
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { | |
// iterate over the images array and add each image to the potential_outputs array | |
for (let j = 0; j < node.images.length; j++) { | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "image", "image": node.images[j], "title": node.title, "node_id": node.id }); | |
} | |
} | |
} | |
else if (node.type === "PreviewImage") { | |
// check if node has an 'images' array property | |
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { | |
// iterate over the images array and add each image to the potential_outputs array | |
for (let j = 0; j < node.images.length; j++) { | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "image", "image": node.images[j], "title": node.title, "node_id": node.id }); | |
} | |
} | |
} | |
else if (node.type === "VHS_VideoCombine") { | |
// check if node has a 'widgets' array property, with type 'image' | |
if (node.hasOwnProperty("widgets") && Array.isArray(node.widgets)) { | |
// iterate over the widgets array and add each image to the potential_outputs array | |
for (let j = 0; j < node.widgets.length; j++) { | |
if (node.widgets[j].type === "image") { | |
const widgetValue = node.widgets[j].value; | |
const parsedURLVals = parseURLPath(widgetValue); | |
// ensure that the parsedURLVals have 'filename', 'subfolder', 'type', and 'format' properties | |
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { | |
if (parsedURLVals.type !== "output") { | |
// TODO | |
} | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "output", 'title': node.title, "node_id": node.id , "output": { "filename": parsedURLVals.filename, "subfolder": parsedURLVals.subfolder, "value": widgetValue, "format": parsedURLVals.format } }); | |
} | |
} else if (node.widgets[j].type === "preview") { | |
const widgetValue = node.widgets[j].value; | |
const parsedURLVals = widgetValue.params; | |
if(!parsedURLVals.format?.startsWith('image')) { | |
// video isn't supported format | |
continue; | |
} | |
// ensure that the parsedURLVals have 'filename', 'subfolder', 'type', and 'format' properties | |
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { | |
if (parsedURLVals.type !== "output") { | |
// TODO | |
} | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "output", 'title': node.title, "node_id": node.id , "output": { "filename": parsedURLVals.filename, "subfolder": parsedURLVals.subfolder, "value": `/view?filename=${parsedURLVals.filename}&subfolder=${parsedURLVals.subfolder}&type=${parsedURLVals.type}&format=${parsedURLVals.format}`, "format": parsedURLVals.format } }); | |
} | |
} | |
} | |
} | |
} | |
else if (node.type === "ADE_AnimateDiffCombine") { | |
// check if node has a 'widgets' array property, with type 'image' | |
if (node.hasOwnProperty("widgets") && Array.isArray(node.widgets)) { | |
// iterate over the widgets array and add each image to the potential_outputs array | |
for (let j = 0; j < node.widgets.length; j++) { | |
if (node.widgets[j].type === "image") { | |
const widgetValue = node.widgets[j].value; | |
const parsedURLVals = parseURLPath(widgetValue); | |
// ensure that the parsedURLVals have 'filename', 'subfolder', 'type', and 'format' properties | |
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { | |
if (parsedURLVals.type !== "output") { | |
// TODO | |
continue; | |
} | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "output", 'title': node.title, "output": { "filename": parsedURLVals.filename, "subfolder": parsedURLVals.subfolder, "type": parsedURLVals.type, "value": widgetValue, "format": parsedURLVals.format } }); | |
} | |
} | |
} | |
} | |
} | |
else if (node.type === "SaveAnimatedWEBP") { | |
// check if node has an 'images' array property | |
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { | |
// iterate over the images array and add each image to the potential_outputs array | |
for (let j = 0; j < node.images.length; j++) { | |
potential_output_nodes.push(node); | |
potential_outputs.push({ "type": "image", "image": node.images[j], "title": node.title }); | |
} | |
} | |
} | |
} | |
// Note: make sure that two arrays are the same length | |
return { potential_outputs, potential_output_nodes }; | |
} | |
export function parseURLPath(urlPath) { | |
// Extract the query string from the URL path | |
var queryString = urlPath.split('?')[1]; | |
// Use the URLSearchParams API to parse the query string | |
var params = new URLSearchParams(queryString); | |
// Create an object to store the parsed parameters | |
var parsedParams = {}; | |
// Iterate over each parameter and add it to the object | |
for (var pair of params.entries()) { | |
parsedParams[pair[0]] = pair[1]; | |
} | |
// Return the object with the parsed parameters | |
return parsedParams; | |
} | |
export const shareToEsheep= () => { | |
app.graphToPrompt() | |
.then(prompt => { | |
const nodes = app.graph._nodes | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
const workflow = prompt['workflow'] | |
api.fetchApi(`/manager/set_esheep_workflow_and_images`, { | |
method: 'POST', | |
headers: { 'Content-Type': 'application/json' }, | |
body: JSON.stringify({ | |
workflow: workflow, | |
images: potential_outputs | |
}) | |
}).then(response => { | |
var domain = window.location.hostname; | |
var port = window.location.port; | |
port = port || (window.location.protocol === 'http:' ? '80' : window.location.protocol === 'https:' ? '443' : ''); | |
var full_domin = domain + ':' + port | |
window.open('https://www.esheep.com/app/workflow_upload?from_local='+ full_domin, '_blank'); | |
}); | |
}) | |
} | |
export const showCopusShareDialog = () => { | |
if (!CopusShareDialog.instance) { | |
CopusShareDialog.instance = new CopusShareDialog(); | |
} | |
return app.graphToPrompt() | |
.then(prompt => { | |
return app.graph._nodes; | |
}) | |
.then(nodes => { | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
CopusShareDialog.instance.show({ potential_outputs, potential_output_nodes}); | |
}) | |
} | |
export const showOpenArtShareDialog = () => { | |
if (!OpenArtShareDialog.instance) { | |
OpenArtShareDialog.instance = new OpenArtShareDialog(); | |
} | |
return app.graphToPrompt() | |
.then(prompt => { | |
// console.log({ prompt }) | |
return app.graph._nodes; | |
}) | |
.then(nodes => { | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
OpenArtShareDialog.instance.show({ potential_outputs, potential_output_nodes}); | |
}) | |
} | |
export const showYouMLShareDialog = () => { | |
if (!YouMLShareDialog.instance) { | |
YouMLShareDialog.instance = new YouMLShareDialog(); | |
} | |
return app.graphToPrompt() | |
.then(prompt => { | |
return app.graph._nodes; | |
}) | |
.then(nodes => { | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
YouMLShareDialog.instance.show(potential_outputs, potential_output_nodes); | |
}) | |
} | |
export const showShareDialog = async (share_option) => { | |
if (!ShareDialog.instance) { | |
ShareDialog.instance = new ShareDialog(share_option); | |
} | |
return app.graphToPrompt() | |
.then(prompt => { | |
// console.log({ prompt }) | |
return app.graph._nodes; | |
}) | |
.then(nodes => { | |
// console.log({ nodes }); | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
if (potential_outputs.length === 0) { | |
if (potential_output_nodes.length === 0) { | |
// todo: add support for other output node types (animatediff combine, etc.) | |
const supported_nodes_string = SUPPORTED_OUTPUT_NODE_TYPES.join(", "); | |
alert(`No supported output node found (${supported_nodes_string}). To share this workflow, please add an output node to your graph and re-run your prompt.`); | |
} else { | |
alert("To share this, first run a prompt. Once it's done, click 'Share'.\n\nNOTE: Images of the Share target can only be selected in the PreviewImage, SaveImage, and VHS_VideoCombine nodes. In the case of VHS_VideoCombine, only the image/gif and image/webp formats are supported."); | |
} | |
return false; | |
} | |
ShareDialog.instance.show({ potential_outputs, potential_output_nodes, share_option }); | |
return true; | |
}); | |
} | |
export class ShareDialogChooser extends ComfyDialog { | |
static instance = null; | |
constructor() { | |
super(); | |
this.element = $el("div.comfy-modal", { | |
parent: document.body, style: { | |
'overflow-y': "auto", | |
} | |
}, | |
[$el("div.comfy-modal-content", | |
{}, | |
[...this.createButtons()]), | |
]); | |
this.selectedNodeId = null; | |
} | |
createButtons() { | |
const buttons = [ | |
{ | |
key: "openart", | |
textContent: "OpenArt AI", | |
website: "https://openart.ai/workflows/", | |
description: "Share ComfyUI workflows and art on OpenArt.ai", | |
onclick: () => { | |
showOpenArtShareDialog(); | |
this.close(); | |
} | |
}, | |
{ | |
key: "youml", | |
textContent: "YouML", | |
website: "https://youml.com", | |
description: "Share your workflow or transform it into an interactive app on YouML.com", | |
onclick: () => { | |
showYouMLShareDialog(); | |
this.close(); | |
} | |
}, | |
{ | |
key: "matrix", | |
textContent: "Matrix Server", | |
website: "https://app.element.io/#/room/%23comfyui_space%3Amatrix.org", | |
description: "Share your art on the official ComfyUI matrix server", | |
onclick: async () => { | |
showShareDialog('matrix').then((suc) => { | |
suc && this.close(); | |
}) | |
} | |
}, | |
{ | |
key: "comfyworkflows", | |
textContent: "ComfyWorkflows", | |
website: "https://comfyworkflows.com", | |
description: "Share & browse thousands of ComfyUI workflows and art 🎨<br/><br/><a style='color:var(--input-text);' href='https://comfyworkflows.com' target='_blank'>ComfyWorkflows.com</a>", | |
onclick: () => { | |
showShareDialog('comfyworkflows').then((suc) => { | |
suc && this.close(); | |
}) | |
} | |
}, | |
{ | |
key: "esheep", | |
textContent: "eSheep", | |
website: "https://www.esheep.com", | |
description: "Share & download thousands of ComfyUI workflows on <a style='color:var(--input-text);' href='https://www.esheep.com' target='_blank'>esheep.com</a>", | |
onclick: () => { | |
shareToEsheep(); | |
this.close(); | |
} | |
}, | |
{ | |
key: "Copus", | |
textContent: "Copus", | |
website: "https://www.copus.io", | |
description: "🔴 Permanently store and secure ownership of your workflow on the open-source platform: <a style='color:var(--input-text);' href='https://copus.io' target='_blank'>Copus.io</a>", | |
onclick: () => { | |
showCopusShareDialog(); | |
this.close(); | |
} | |
}, | |
]; | |
function createShareButtonsWithDescriptions() { | |
// Responsive container | |
const container = $el("div", { | |
style: { | |
display: "flex", | |
'flex-wrap': 'wrap', | |
'justify-content': 'space-around', | |
'padding': '10px', | |
} | |
}); | |
buttons.forEach(b => { | |
const button = $el("button", { | |
type: "button", | |
textContent: b.textContent, | |
onclick: b.onclick, | |
style: { | |
'width': '25%', | |
'minWidth': '200px', | |
'background-color': b.backgroundColor || '', | |
'border-radius': '5px', | |
'cursor': 'pointer', | |
'padding': '5px 5px', | |
'margin-bottom': '5px', | |
'transition': 'background-color 0.3s', | |
} | |
}); | |
button.addEventListener('mouseover', () => { | |
button.style.backgroundColor = '#007BFF'; // Change color on hover | |
}); | |
button.addEventListener('mouseout', () => { | |
button.style.backgroundColor = b.backgroundColor || ''; | |
}); | |
const description = $el("p", { | |
innerHTML: b.description, | |
style: { | |
'text-align': 'left', | |
color: 'var(--input-text)', | |
'font-size': '14px', | |
'margin-bottom': '0', | |
}, | |
}); | |
const websiteLink = $el("a", { | |
textContent: "🌐 Website", | |
href: b.website, | |
target: "_blank", | |
style: { | |
color: 'var(--input-text)', | |
'margin-left': '10px', | |
'font-size': '12px', | |
'text-decoration': 'none', | |
'align-self': 'center', | |
}, | |
}); | |
// Add highlight to the website link | |
websiteLink.addEventListener('mouseover', () => { | |
websiteLink.style.opacity = '0.7'; | |
}); | |
websiteLink.addEventListener('mouseout', () => { | |
websiteLink.style.opacity = '1'; | |
}); | |
const buttonLinkContainer = $el("div", { | |
style: { | |
display: 'flex', | |
'align-items': 'center', | |
'margin-bottom': '10px', | |
} | |
}, [button, websiteLink]); | |
const column = $el("div", { | |
style: { | |
'flex-basis': '100%', | |
'margin': '10px', | |
'padding': '10px 20px', | |
'border': '1px solid #ddd', | |
'border-radius': '5px', | |
'box-shadow': '0 2px 4px rgba(0, 0, 0, 0.1)', | |
} | |
}, [buttonLinkContainer, description]); | |
container.appendChild(column); | |
}); | |
return container; | |
} | |
return [ | |
$el("p", { | |
textContent: 'Choose a platform to share your workflow', | |
style: { | |
'text-align': 'center', | |
'color': 'var(--input-text)', | |
'font-size': '18px', | |
'margin-bottom': '10px', | |
}, | |
} | |
), | |
$el("div.cm-menu-container", { | |
id: "comfyui-share-container" | |
}, [ | |
$el("div.cm-menu-column", [ | |
createShareButtonsWithDescriptions(), | |
$el("br", {}, []), | |
]), | |
]), | |
$el("div.cm-menu-container", { | |
id: "comfyui-share-container" | |
}, [ | |
$el("button", { | |
type: "button", | |
style: { | |
margin: "0 25px", | |
width: "100%", | |
}, | |
textContent: "Close", | |
onclick: () => { | |
this.close() | |
} | |
}), | |
$el("br", {}, []), | |
]), | |
]; | |
} | |
show() { | |
this.element.style.display = "block"; | |
this.element.style.zIndex = 10001; | |
} | |
} | |
export class ShareDialog extends ComfyDialog { | |
static instance = null; | |
static matrix_auth = { homeserver: "matrix.org", username: "", password: "" }; | |
static cw_sharekey = ""; | |
constructor(share_option) { | |
super(); | |
this.share_option = share_option; | |
this.element = $el("div.comfy-modal", { | |
parent: document.body, style: { | |
'overflow-y': "auto", | |
} | |
}, | |
[$el("div.comfy-modal-content", | |
{}, | |
[...this.createButtons()]), | |
]); | |
this.selectedOutputIndex = 0; | |
} | |
createButtons() { | |
this.radio_buttons = $el("div", { | |
id: "selectOutputImages", | |
}, []); | |
this.is_nsfw_checkbox = $el("input", { type: 'checkbox', id: "is_nsfw" }, []) | |
const is_nsfw_checkbox_text = $el("label", { | |
}, [" Is this NSFW?"]) | |
this.is_nsfw_checkbox.style.color = "var(--fg-color)"; | |
this.is_nsfw_checkbox.checked = false; | |
this.matrix_destination_checkbox = $el("input", { type: 'checkbox', id: "matrix_destination" }, []) | |
const matrix_destination_checkbox_text = $el("label", {}, [" ComfyUI Matrix server"]) | |
this.matrix_destination_checkbox.style.color = "var(--fg-color)"; | |
this.matrix_destination_checkbox.checked = this.share_option === 'matrix'; //true; | |
this.comfyworkflows_destination_checkbox = $el("input", { type: 'checkbox', id: "comfyworkflows_destination" }, []) | |
const comfyworkflows_destination_checkbox_text = $el("label", {}, [" ComfyWorkflows.com"]) | |
this.comfyworkflows_destination_checkbox.style.color = "var(--fg-color)"; | |
this.comfyworkflows_destination_checkbox.checked = this.share_option !== 'matrix'; | |
this.matrix_homeserver_input = $el("input", { type: 'text', id: "matrix_homeserver", placeholder: "matrix.org", value: ShareDialog.matrix_auth.homeserver || 'matrix.org' }, []); | |
this.matrix_username_input = $el("input", { type: 'text', placeholder: "Username", value: ShareDialog.matrix_auth.username || '' }, []); | |
this.matrix_password_input = $el("input", { type: 'password', placeholder: "Password", value: ShareDialog.matrix_auth.password || '' }, []); | |
this.cw_sharekey_input = $el("input", { type: 'text', placeholder: "Share key (found on your profile page)", value: ShareDialog.cw_sharekey || '' }, []); | |
this.cw_sharekey_input.style.width = "100%"; | |
this.credits_input = $el("input", { | |
type: "text", | |
placeholder: "This will be used to give credits", | |
required: false, | |
}, []); | |
this.title_input = $el("input", { | |
type: "text", | |
placeholder: "ex: My awesome art", | |
required: false | |
}, []); | |
this.description_input = $el("textarea", { | |
placeholder: "ex: Trying out a new workflow... ", | |
required: false, | |
}, []); | |
this.share_button = $el("button", { | |
type: "submit", | |
textContent: "Share", | |
style: { | |
backgroundColor: "blue" | |
} | |
}, []); | |
this.final_message = $el("div", { | |
style: { | |
color: "white", | |
textAlign: "center", | |
// marginTop: "10px", | |
// backgroundColor: "black", | |
padding: "10px", | |
} | |
}, []); | |
this.share_finalmessage_container = $el("div.cm-menu-container", { | |
id: "comfyui-share-finalmessage-container", | |
style: { | |
display: "none", | |
} | |
}, [ | |
$el("div.cm-menu-column", [ | |
this.final_message, | |
$el("button", { | |
type: "button", | |
textContent: "Close", | |
onclick: () => { | |
// Reset state | |
this.matrix_destination_checkbox.checked = this.share_option === 'matrix'; | |
this.comfyworkflows_destination_checkbox.checked = this.share_option !== 'matrix'; | |
this.share_button.textContent = "Share"; | |
this.share_button.style.display = "inline-block"; | |
this.final_message.innerHTML = ""; | |
this.final_message.style.color = "white"; | |
this.credits_input.value = ""; | |
this.title_input.value = ""; | |
this.description_input.value = ""; | |
this.is_nsfw_checkbox.checked = false; | |
this.selectedOutputIndex = 0; | |
// hide the final message | |
this.share_finalmessage_container.style.display = "none"; | |
// show the share container | |
this.share_container.style.display = "flex"; | |
this.close() | |
} | |
}), | |
]) | |
]); | |
this.share_container = $el("div.cm-menu-container", { | |
id: "comfyui-share-container" | |
}, [ | |
$el("div.cm-menu-column", [ | |
$el("details", { | |
style: { | |
border: "1px solid #999", | |
padding: "5px", | |
borderRadius: "5px", | |
backgroundColor: "#222" | |
} | |
}, [ | |
$el("summary", { | |
style: { | |
color: "white", | |
cursor: "pointer", | |
} | |
}, [`Matrix account`]), | |
$el("div", { | |
style: { | |
display: "flex", | |
flexDirection: "row", | |
} | |
}, [ | |
$el("div", { | |
textContent: "Homeserver", | |
style: { | |
marginRight: "10px", | |
} | |
}, []), | |
this.matrix_homeserver_input, | |
]), | |
$el("div", { | |
style: { | |
display: "flex", | |
flexDirection: "row", | |
} | |
}, [ | |
$el("div", { | |
textContent: "Username", | |
style: { | |
marginRight: "10px", | |
} | |
}, []), | |
this.matrix_username_input, | |
]), | |
$el("div", { | |
style: { | |
display: "flex", | |
flexDirection: "row", | |
} | |
}, [ | |
$el("div", { | |
textContent: "Password", | |
style: { | |
marginRight: "10px", | |
} | |
}, []), | |
this.matrix_password_input, | |
]), | |
]), | |
$el("details", { | |
style: { | |
border: "1px solid #999", | |
marginTop: "10px", | |
padding: "5px", | |
borderRadius: "5px", | |
backgroundColor: "#222" | |
}, | |
}, [ | |
$el("summary", { | |
style: { | |
color: "white", | |
cursor: "pointer", | |
} | |
}, [`Comfyworkflows.com account`]), | |
$el("h4", { | |
textContent: "Share key (found on your profile page)", | |
}, []), | |
$el("p", { size: 3, color: "white" }, ["If provided, your art will be saved to your account. Otherwise, it will be shared anonymously."]), | |
this.cw_sharekey_input, | |
]), | |
$el("div", {}, [ | |
$el("p", { | |
size: 3, color: "white", style: { | |
color: 'var(--input-text)' | |
} | |
}, [`Select where to share your art:`]), | |
this.matrix_destination_checkbox, | |
matrix_destination_checkbox_text, | |
$el("br", {}, []), | |
this.comfyworkflows_destination_checkbox, | |
comfyworkflows_destination_checkbox_text, | |
]), | |
$el("h4", { | |
textContent: "Credits (optional)", | |
size: 3, | |
color: "white", | |
style: { | |
color: 'var(--input-text)' | |
} | |
}, []), | |
this.credits_input, | |
// $el("br", {}, []), | |
$el("h4", { | |
textContent: "Title (optional)", | |
size: 3, | |
color: "white", | |
style: { | |
color: 'var(--input-text)' | |
} | |
}, []), | |
this.title_input, | |
// $el("br", {}, []), | |
$el("h4", { | |
textContent: "Description (optional)", | |
size: 3, | |
color: "white", | |
style: { | |
color: 'var(--input-text)' | |
} | |
}, []), | |
this.description_input, | |
$el("br", {}, []), | |
$el("div", {}, [this.is_nsfw_checkbox, is_nsfw_checkbox_text]), | |
// $el("br", {}, []), | |
// this.final_message, | |
// $el("br", {}, []), | |
]), | |
$el("div.cm-menu-column", [ | |
this.radio_buttons, | |
$el("br", {}, []), | |
this.share_button, | |
$el("button", { | |
type: "button", | |
textContent: "Close", | |
onclick: () => { | |
// Reset state | |
this.matrix_destination_checkbox.checked = this.share_option === 'matrix'; | |
this.comfyworkflows_destination_checkbox.checked = this.share_option !== 'matrix'; | |
this.share_button.textContent = "Share"; | |
this.share_button.style.display = "inline-block"; | |
this.final_message.innerHTML = ""; | |
this.final_message.style.color = "white"; | |
this.credits_input.value = ""; | |
this.title_input.value = ""; | |
this.description_input.value = ""; | |
this.is_nsfw_checkbox.checked = false; | |
this.selectedOutputIndex = 0; | |
// hide the final message | |
this.share_finalmessage_container.style.display = "none"; | |
// show the share container | |
this.share_container.style.display = "flex"; | |
this.close() | |
} | |
}), | |
$el("br", {}, []), | |
]), | |
]); | |
// get the user's existing matrix auth and share key | |
ShareDialog.matrix_auth = { homeserver: "matrix.org", username: "", password: "" }; | |
try { | |
api.fetchApi(`/manager/get_matrix_auth`) | |
.then(response => response.json()) | |
.then(data => { | |
ShareDialog.matrix_auth = data; | |
this.matrix_homeserver_input.value = ShareDialog.matrix_auth.homeserver; | |
this.matrix_username_input.value = ShareDialog.matrix_auth.username; | |
this.matrix_password_input.value = ShareDialog.matrix_auth.password; | |
}) | |
.catch(error => { | |
// console.log(error); | |
}); | |
} catch (error) { | |
// console.log(error); | |
} | |
// get the user's existing comfyworkflows share key | |
ShareDialog.cw_sharekey = ""; | |
try { | |
// console.log("Fetching comfyworkflows share key") | |
api.fetchApi(`/manager/get_comfyworkflows_auth`) | |
.then(response => response.json()) | |
.then(data => { | |
ShareDialog.cw_sharekey = data.comfyworkflows_sharekey; | |
this.cw_sharekey_input.value = ShareDialog.cw_sharekey; | |
}) | |
.catch(error => { | |
// console.log(error); | |
}); | |
} catch (error) { | |
// console.log(error); | |
} | |
this.share_button.onclick = async () => { | |
const prompt = await app.graphToPrompt(); | |
const nodes = app.graph._nodes; | |
// console.log({ prompt, nodes }); | |
const destinations = []; | |
if (this.matrix_destination_checkbox.checked) { | |
destinations.push("matrix"); | |
} | |
if (this.comfyworkflows_destination_checkbox.checked) { | |
destinations.push("comfyworkflows"); | |
} | |
// if destinations includes matrix, make an api call to /manager/check_matrix to ensure that the user has configured their matrix settings | |
if (destinations.includes("matrix")) { | |
let definedMatrixAuth = !!this.matrix_homeserver_input.value && !!this.matrix_username_input.value && !!this.matrix_password_input.value; | |
if (!definedMatrixAuth) { | |
alert("Please set your Matrix account details."); | |
return; | |
} | |
} | |
if (destinations.includes("comfyworkflows") && !this.cw_sharekey_input.value && false) { //!confirm("You have NOT set your ComfyWorkflows.com share key. Your art will NOT be connected to your account (it will be shared anonymously). Continue?")) { | |
return; | |
} | |
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); | |
// console.log({ potential_outputs, potential_output_nodes }) | |
if (potential_outputs.length === 0) { | |
if (potential_output_nodes.length === 0) { | |
// todo: add support for other output node types (animatediff combine, etc.) | |
const supported_nodes_string = SUPPORTED_OUTPUT_NODE_TYPES.join(", "); | |
alert(`No supported output node found (${supported_nodes_string}). To share this workflow, please add an output node to your graph and re-run your prompt.`); | |
} else { | |
alert("To share this, first run a prompt. Once it's done, click 'Share'.\n\nNOTE: Images of the Share target can only be selected in the PreviewImage, SaveImage, and VHS_VideoCombine nodes. In the case of VHS_VideoCombine, only the image/gif and image/webp formats are supported."); | |
} | |
this.selectedOutputIndex = 0; | |
this.close(); | |
return; | |
} | |
// Change the text of the share button to "Sharing..." to indicate that the share process has started | |
this.share_button.textContent = "Sharing..."; | |
const response = await api.fetchApi(`/manager/share`, { | |
method: 'POST', | |
headers: { 'Content-Type': 'application/json' }, | |
body: JSON.stringify({ | |
matrix_auth: { | |
homeserver: this.matrix_homeserver_input.value, | |
username: this.matrix_username_input.value, | |
password: this.matrix_password_input.value, | |
}, | |
cw_auth: { | |
cw_sharekey: this.cw_sharekey_input.value, | |
}, | |
share_destinations: destinations, | |
credits: this.credits_input.value, | |
title: this.title_input.value, | |
description: this.description_input.value, | |
is_nsfw: this.is_nsfw_checkbox.checked, | |
prompt, | |
potential_outputs, | |
selected_output_index: this.selectedOutputIndex, | |
// potential_output_nodes | |
}) | |
}); | |
if (response.status != 200) { | |
try { | |
const response_json = await response.json(); | |
if (response_json.error) { | |
alert(response_json.error); | |
this.close(); | |
return; | |
} else { | |
alert("Failed to share your art. Please try again."); | |
this.close(); | |
return; | |
} | |
} catch (e) { | |
alert("Failed to share your art. Please try again."); | |
this.close(); | |
return; | |
} | |
} | |
const response_json = await response.json(); | |
if (response_json.comfyworkflows.url) { | |
this.final_message.innerHTML = "Your art has been shared: <a href='" + response_json.comfyworkflows.url + "' target='_blank'>" + response_json.comfyworkflows.url + "</a>"; | |
if (response_json.matrix.success) { | |
this.final_message.innerHTML += "<br>Your art has been shared in the ComfyUI Matrix server's #share channel!"; | |
} | |
} else { | |
if (response_json.matrix.success) { | |
this.final_message.innerHTML = "Your art has been shared in the ComfyUI Matrix server's #share channel!"; | |
} | |
} | |
this.final_message.style.color = "green"; | |
// hide #comfyui-share-container and show #comfyui-share-finalmessage-container | |
this.share_container.style.display = "none"; | |
this.share_finalmessage_container.style.display = "block"; | |
// hide the share button | |
this.share_button.textContent = "Shared!"; | |
this.share_button.style.display = "none"; | |
// this.close(); | |
} | |
const res = | |
[ | |
$el("tr.td", { width: "100%" }, [ | |
$el("font", { size: 6, color: "white" }, [`Share your art`]), | |
]), | |
$el("br", {}, []), | |
this.share_finalmessage_container, | |
this.share_container, | |
]; | |
res[0].style.padding = "10px 10px 10px 10px"; | |
res[0].style.backgroundColor = "black"; //"linear-gradient(90deg, #00C9FF 0%, #92FE9D 100%)"; | |
res[0].style.textAlign = "center"; | |
res[0].style.height = "45px"; | |
return res; | |
} | |
show({potential_outputs, potential_output_nodes, share_option}) { | |
// Sort `potential_output_nodes` by node ID to make the order always | |
// consistent, but we should also keep `potential_outputs` in the same | |
// order as `potential_output_nodes`. | |
const potential_output_to_order = {}; | |
potential_output_nodes.forEach((node, index) => { | |
if (node.id in potential_output_to_order) { | |
potential_output_to_order[node.id][1].push(potential_outputs[index]); | |
} else { | |
potential_output_to_order[node.id] = [node, [potential_outputs[index]]]; | |
} | |
}) | |
// Sort the object `potential_output_to_order` by key (node ID) | |
const sorted_potential_output_to_order = Object.fromEntries( | |
Object.entries(potential_output_to_order).sort((a, b) => a[0].id - b[0].id) | |
); | |
const sorted_potential_outputs = [] | |
const sorted_potential_output_nodes = [] | |
for (const [key, value] of Object.entries(sorted_potential_output_to_order)) { | |
sorted_potential_output_nodes.push(value[0]); | |
sorted_potential_outputs.push(...value[1]); | |
} | |
potential_output_nodes = sorted_potential_output_nodes; | |
potential_outputs = sorted_potential_outputs; | |
// console.log({ potential_outputs, potential_output_nodes }) | |
this.radio_buttons.innerHTML = ""; // clear the radio buttons | |
let is_radio_button_checked = false; // only check the first radio button if multiple images from the same node | |
const new_radio_buttons = $el("div", { | |
id: "selectOutput-Options", | |
style: { | |
'overflow-y': 'scroll', | |
'max-height': '400px', | |
} | |
}, potential_outputs.map((output, index) => { | |
const {node_id} = output; | |
const radio_button = $el("input", { type: 'radio', name: "selectOutputImages", value: index, required: index === 0 }, []) | |
let radio_button_img; | |
if (output.type === "image" || output.type === "temp") { | |
radio_button_img = $el("img", { src: `/view?filename=${output.image.filename}&subfolder=${output.image.subfolder}&type=${output.image.type}`, style: { width: "auto", height: "100px" } }, []); | |
} else if (output.type === "output") { | |
radio_button_img = $el("img", { src: output.output.value, style: { width: "auto", height: "100px" } }, []); | |
} else { | |
// unsupported output type | |
// this should never happen | |
// TODO | |
radio_button_img = $el("img", { src: "", style: { width: "auto", height: "100px" } }, []); | |
} | |
const radio_button_text = $el("label", { | |
// style: { | |
// color: 'var(--input-text)' | |
// } | |
}, [output.title]) | |
radio_button.style.color = "var(--fg-color)"; | |
// Make the radio button checked if it's the selected node, | |
// otherwise make the first radio button checked. | |
if (this.selectedNodeId) { | |
if (this.selectedNodeId === node_id && !is_radio_button_checked) { | |
radio_button.checked = true; | |
is_radio_button_checked = true; | |
} | |
} else { | |
radio_button.checked = index === 0; | |
} | |
if (radio_button.checked) { | |
this.selectedOutputIndex = index; | |
} | |
radio_button.onchange = () => { | |
this.selectedOutputIndex = parseInt(radio_button.value); | |
}; | |
return $el("div", { | |
style: { | |
display: "flex", | |
'align-items': 'center', | |
'justify-content': 'space-between', | |
'margin-bottom': '10px', | |
} | |
}, [radio_button, radio_button_text, radio_button_img]); | |
})); | |
const header = $el("h3", { | |
textContent: "Select an image to share", | |
size: 3, | |
color: "white", | |
style: { | |
'text-align': 'center', | |
color: 'var(--input-text)', | |
backgroundColor: 'black', | |
padding: '10px', | |
'margin-top': '0px', | |
} | |
}, [ | |
$el("p", { | |
textContent: "Scroll to see all outputs", | |
size: 2, | |
color: "white", | |
style: { | |
'text-align': 'center', | |
color: 'var(--input-text)', | |
'margin-bottom': '5px', | |
'font-style': 'italic', | |
'font-size': '12px', | |
}, | |
}, []) | |
]); | |
this.radio_buttons.appendChild(header); | |
// this.radio_buttons.appendChild(subheader); | |
this.radio_buttons.appendChild(new_radio_buttons); | |
this.element.style.display = "block"; | |
share_option = share_option || this.share_option; | |
if (share_option === 'comfyworkflows') { | |
this.matrix_destination_checkbox.checked = false; | |
this.comfyworkflows_destination_checkbox.checked = true; | |
} else { | |
this.matrix_destination_checkbox.checked = true; | |
this.comfyworkflows_destination_checkbox.checked = false; | |
} | |
} | |
} | |