| 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"; |
| import { customAlert } from "./common.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(", "); |
| customAlert(`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 { |
| customAlert("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: "🔴 Earn simple. Get paid from your ComfyUI workflows—no revenue sharing. Ever.", |
| 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', |
| 'position':'relative' |
| } |
| }, |
| [ |
| $el("span", { style: { |
| } }), |
| ] |
| ); |
| 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 copus_ui =$el("div", { style: { |
| 'position': 'absolute', |
| 'height': '100%', |
| 'left': '-25px', |
| 'top': '-26px', |
| 'width': '100%', |
| 'z-index':'-1', |
| 'background':'url("https://static.copus.io/images/client/202412/test/f28ac6ef8f4c6f3d5d50856a272ed02c.png")', |
| 'background-repeat': 'no-repeat', |
| } }); |
| const copus_ui_bottom =$el("div", { style: { |
| 'position': 'absolute', |
| 'height': '100%', |
| 'left': '25px', |
| 'bottom': '-26px', |
| 'width': '100%', |
| 'transform':'scale(-1, -1)', |
| 'z-index':'-1', |
| 'background':'url("https://static.copus.io/images/client/202412/test/f28ac6ef8f4c6f3d5d50856a272ed02c.png")', |
| 'background-repeat': 'no-repeat', |
| } }); |
|
|
| 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)', |
| 'position':'relative' |
| } |
| }, [buttonLinkContainer, description |
| , |
| b.key ==='Copus' ? |
| copus_ui |
| :'', |
| b.key ==='Copus' ? |
| copus_ui_bottom |
| :'', |
| ]); |
|
|
| 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 = 1099; |
| } |
| } |
| 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) { |
| customAlert("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(", "); |
| customAlert(`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 { |
| customAlert("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) { |
| customAlert(response_json.error); |
| this.close(); |
| return; |
| } else { |
| customAlert("Failed to share your art. Please try again."); |
| this.close(); |
| return; |
| } |
| } catch (e) { |
| customAlert("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; |
| } |
| } |
| } |
|
|