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 🎨

ComfyWorkflows.com", 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 esheep.com", 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: Copus.io", 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: " + response_json.comfyworkflows.url + ""; if (response_json.matrix.success) { this.final_message.innerHTML += "
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; } } }