|
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 = []; |
|
|
|
|
|
|
|
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") { |
|
|
|
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { |
|
|
|
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") { |
|
|
|
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { |
|
|
|
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") { |
|
|
|
if (node.hasOwnProperty("widgets") && Array.isArray(node.widgets)) { |
|
|
|
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); |
|
|
|
|
|
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { |
|
if (parsedURLVals.type !== "output") { |
|
|
|
} |
|
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')) { |
|
|
|
continue; |
|
} |
|
|
|
|
|
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { |
|
if (parsedURLVals.type !== "output") { |
|
|
|
} |
|
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") { |
|
|
|
if (node.hasOwnProperty("widgets") && Array.isArray(node.widgets)) { |
|
|
|
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); |
|
|
|
if (parsedURLVals.hasOwnProperty("filename") && parsedURLVals.hasOwnProperty("subfolder") && parsedURLVals.hasOwnProperty("type") && parsedURLVals.hasOwnProperty("format")) { |
|
if (parsedURLVals.type !== "output") { |
|
|
|
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") { |
|
|
|
if (node.hasOwnProperty("images") && Array.isArray(node.images)) { |
|
|
|
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 }); |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
return { potential_outputs, potential_output_nodes }; |
|
} |
|
|
|
|
|
export function parseURLPath(urlPath) { |
|
|
|
var queryString = urlPath.split('?')[1]; |
|
|
|
|
|
var params = new URLSearchParams(queryString); |
|
|
|
|
|
var parsedParams = {}; |
|
|
|
|
|
for (var pair of params.entries()) { |
|
parsedParams[pair[0]] = pair[1]; |
|
} |
|
|
|
|
|
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 => { |
|
|
|
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 => { |
|
|
|
return app.graph._nodes; |
|
}) |
|
.then(nodes => { |
|
|
|
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); |
|
if (potential_outputs.length === 0) { |
|
if (potential_output_nodes.length === 0) { |
|
|
|
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() { |
|
|
|
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'; |
|
}); |
|
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', |
|
}, |
|
}); |
|
|
|
|
|
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'; |
|
|
|
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", |
|
|
|
|
|
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: () => { |
|
|
|
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; |
|
|
|
|
|
this.share_finalmessage_container.style.display = "none"; |
|
|
|
|
|
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("h4", { |
|
textContent: "Title (optional)", |
|
size: 3, |
|
color: "white", |
|
style: { |
|
color: 'var(--input-text)' |
|
} |
|
}, []), |
|
this.title_input, |
|
|
|
|
|
$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("div.cm-menu-column", [ |
|
this.radio_buttons, |
|
$el("br", {}, []), |
|
|
|
this.share_button, |
|
|
|
$el("button", { |
|
type: "button", |
|
textContent: "Close", |
|
onclick: () => { |
|
|
|
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; |
|
|
|
|
|
this.share_finalmessage_container.style.display = "none"; |
|
|
|
|
|
this.share_container.style.display = "flex"; |
|
|
|
this.close() |
|
} |
|
}), |
|
$el("br", {}, []), |
|
]), |
|
]); |
|
|
|
|
|
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 => { |
|
|
|
}); |
|
} catch (error) { |
|
|
|
} |
|
|
|
|
|
ShareDialog.cw_sharekey = ""; |
|
try { |
|
|
|
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 => { |
|
|
|
}); |
|
} catch (error) { |
|
|
|
} |
|
|
|
this.share_button.onclick = async () => { |
|
const prompt = await app.graphToPrompt(); |
|
const nodes = app.graph._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")) { |
|
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) { |
|
return; |
|
} |
|
|
|
const { potential_outputs, potential_output_nodes } = getPotentialOutputsAndOutputNodes(nodes); |
|
|
|
|
|
|
|
if (potential_outputs.length === 0) { |
|
if (potential_output_nodes.length === 0) { |
|
|
|
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; |
|
} |
|
|
|
|
|
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, |
|
|
|
}) |
|
}); |
|
|
|
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"; |
|
|
|
|
|
this.share_container.style.display = "none"; |
|
this.share_finalmessage_container.style.display = "block"; |
|
|
|
|
|
this.share_button.textContent = "Shared!"; |
|
this.share_button.style.display = "none"; |
|
|
|
} |
|
|
|
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"; |
|
res[0].style.textAlign = "center"; |
|
res[0].style.height = "45px"; |
|
return res; |
|
} |
|
|
|
show({potential_outputs, potential_output_nodes, share_option}) { |
|
|
|
|
|
|
|
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]]]; |
|
} |
|
}) |
|
|
|
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; |
|
|
|
|
|
this.radio_buttons.innerHTML = ""; |
|
let is_radio_button_checked = false; |
|
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 { |
|
|
|
|
|
|
|
radio_button_img = $el("img", { src: "", style: { width: "auto", height: "100px" } }, []); |
|
} |
|
const radio_button_text = $el("label", { |
|
|
|
|
|
|
|
}, [output.title]) |
|
radio_button.style.color = "var(--fg-color)"; |
|
|
|
|
|
|
|
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(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; |
|
} |
|
} |
|
} |
|
|