Spaces:
Configuration error
Configuration error
| import { api } from "../../../../scripts/api.js"; | |
| import { app } from "../../../../scripts/app.js"; | |
| import {deepEqual, addCss, addMeta, isLocalNetwork} from "../common/utils.js"; | |
| import {logoIcon, quesitonIcon, rocketIcon, groupIcon, rebootIcon, closeIcon} from "../common/icon.js"; | |
| import {$t} from '../common/i18n.js'; | |
| import {toast} from "../common/toast.js"; | |
| import {$el, ComfyDialog} from "../../../../scripts/ui.js"; | |
| addCss('css/index.css') | |
| api.addEventListener("easyuse-toast",event=>{ | |
| const content = event.detail.content | |
| const type = event.detail.type | |
| const duration = event.detail.duration | |
| if(!type){ | |
| toast.info(content, duration) | |
| } | |
| else{ | |
| toast.showToast({ | |
| id: `toast-${type}`, | |
| content: `${toast[type+"_icon"]} ${content}`, | |
| duration: duration || 3000, | |
| }) | |
| } | |
| }) | |
| let draggerEl = null | |
| let isGroupMapcanMove = true | |
| function createGroupMap(){ | |
| let div = document.querySelector('#easyuse_groups_map') | |
| if(div){ | |
| div.style.display = div.style.display == 'none' ? 'flex' : 'none' | |
| return | |
| } | |
| let groups = app.canvas.graph._groups | |
| let nodes = app.canvas.graph._nodes | |
| let old_nodes = groups.length | |
| div = document.createElement('div') | |
| div.id = 'easyuse_groups_map' | |
| div.innerHTML = '' | |
| let btn = document.createElement('div') | |
| btn.style = `display: flex; | |
| width: calc(100% - 8px); | |
| justify-content: space-between; | |
| align-items: center; | |
| padding: 0 6px; | |
| height: 44px;` | |
| let hideBtn = $el('button.closeBtn',{ | |
| innerHTML:closeIcon, | |
| onclick:_=>div.style.display = 'none' | |
| }) | |
| let textB = document.createElement('p') | |
| btn.appendChild(textB) | |
| btn.appendChild(hideBtn) | |
| textB.style.fontSize = '11px' | |
| textB.innerHTML = `<b>${$t('Groups Map')} (EasyUse)</b>` | |
| div.appendChild(btn) | |
| div.addEventListener('mousedown', function (e) { | |
| var startX = e.clientX | |
| var startY = e.clientY | |
| var offsetX = div.offsetLeft | |
| var offsetY = div.offsetTop | |
| function moveBox (e) { | |
| var newX = e.clientX | |
| var newY = e.clientY | |
| var deltaX = newX - startX | |
| var deltaY = newY - startY | |
| div.style.left = offsetX + deltaX + 'px' | |
| div.style.top = offsetY + deltaY + 'px' | |
| } | |
| function stopMoving () { | |
| document.removeEventListener('mousemove', moveBox) | |
| document.removeEventListener('mouseup', stopMoving) | |
| } | |
| if(isGroupMapcanMove){ | |
| document.addEventListener('mousemove', moveBox) | |
| document.addEventListener('mouseup', stopMoving) | |
| } | |
| }) | |
| function updateGroups(groups, groupsDiv, autoSortDiv){ | |
| if(groups.length>0){ | |
| autoSortDiv.style.display = 'block' | |
| }else autoSortDiv.style.display = 'none' | |
| for (let index in groups) { | |
| const group = groups[index] | |
| const title = group.title | |
| const show_text = $t('Always') | |
| const hide_text = $t('Bypass') | |
| const mute_text = $t('Never') | |
| let group_item = document.createElement('div') | |
| let group_item_style = `justify-content: space-between;display:flex;background-color: var(--comfy-input-bg);border-radius: 5px;border:1px solid var(--border-color);margin-top:5px;` | |
| group_item.addEventListener("mouseover",event=>{ | |
| event.preventDefault() | |
| group_item.style = group_item_style + "filter:brightness(1.2);" | |
| }) | |
| group_item.addEventListener("mouseleave",event=>{ | |
| event.preventDefault() | |
| group_item.style = group_item_style + "filter:brightness(1);" | |
| }) | |
| group_item.addEventListener("dragstart",e=>{ | |
| draggerEl = e.currentTarget; | |
| e.currentTarget.style.opacity = "0.6"; | |
| e.currentTarget.style.border = "1px dashed yellow"; | |
| e.dataTransfer.effectAllowed = 'move'; | |
| e.dataTransfer.setDragImage(emptyImg, 0, 0); | |
| }) | |
| group_item.addEventListener("dragend",e=>{ | |
| e.target.style.opacity = "1"; | |
| e.currentTarget.style.border = "1px dashed transparent"; | |
| e.currentTarget.removeAttribute("draggable"); | |
| document.querySelectorAll('.easyuse-group-item').forEach((el,i) => { | |
| var prev_i = el.dataset.id; | |
| if (el == draggerEl && prev_i != i ) { | |
| groups.splice(i, 0, groups.splice(prev_i, 1)[0]); | |
| } | |
| el.dataset.id = i; | |
| }); | |
| isGroupMapcanMove = true | |
| }) | |
| group_item.addEventListener("dragover",e=>{ | |
| e.preventDefault(); | |
| if (e.currentTarget == draggerEl) return; | |
| let rect = e.currentTarget.getBoundingClientRect(); | |
| if (e.clientY > rect.top + rect.height / 2) { | |
| e.currentTarget.parentNode.insertBefore(draggerEl, e.currentTarget.nextSibling); | |
| } else { | |
| e.currentTarget.parentNode.insertBefore(draggerEl, e.currentTarget); | |
| } | |
| isGroupMapcanMove = true | |
| }) | |
| group_item.setAttribute('data-id',index) | |
| group_item.className = 'easyuse-group-item' | |
| group_item.style = group_item_style | |
| // 标题 | |
| let text_group_title = document.createElement('div') | |
| text_group_title.style = `flex:1;font-size:12px;color:var(--input-text);padding:4px;white-space: nowrap;overflow: hidden;text-overflow: ellipsis;cursor:pointer` | |
| text_group_title.innerHTML = `${title}` | |
| text_group_title.addEventListener('mousedown',e=>{ | |
| isGroupMapcanMove = false | |
| e.currentTarget.parentNode.draggable = 'true'; | |
| }) | |
| text_group_title.addEventListener('mouseleave',e=>{ | |
| setTimeout(_=>{ | |
| isGroupMapcanMove = true | |
| },150) | |
| }) | |
| group_item.append(text_group_title) | |
| // 按钮组 | |
| let buttons = document.createElement('div') | |
| group.recomputeInsideNodes(); | |
| const nodesInGroup = group._nodes; | |
| let isGroupShow = nodesInGroup && nodesInGroup.length>0 && nodesInGroup[0].mode == 0 | |
| let isGroupMute = nodesInGroup && nodesInGroup.length>0 && nodesInGroup[0].mode == 2 | |
| let go_btn = document.createElement('button') | |
| go_btn.style = "margin-right:6px;cursor:pointer;font-size:10px;padding:2px 4px;color:var(--input-text);background-color: var(--comfy-input-bg);border: 1px solid var(--border-color);border-radius:4px;" | |
| go_btn.innerText = "Go" | |
| go_btn.addEventListener('click', () => { | |
| app.canvas.ds.offset[0] = -group.pos[0] - group.size[0] * 0.5 + (app.canvas.canvas.width * 0.5) / app.canvas.ds.scale; | |
| app.canvas.ds.offset[1] = -group.pos[1] - group.size[1] * 0.5 + (app.canvas.canvas.height * 0.5) / app.canvas.ds.scale; | |
| app.canvas.setDirty(true, true); | |
| app.canvas.setZoom(1) | |
| }) | |
| buttons.append(go_btn) | |
| let see_btn = document.createElement('button') | |
| let defaultStyle = `cursor:pointer;font-size:10px;;padding:2px;border: 1px solid var(--border-color);border-radius:4px;width:36px;` | |
| see_btn.style = isGroupMute ? `background-color:var(--error-text);color:var(--input-text);` + defaultStyle : (isGroupShow ? `background-color:var(--theme-color);color:var(--input-text);` + defaultStyle : `background-color: var(--comfy-input-bg);color:var(--descrip-text);` + defaultStyle) | |
| see_btn.innerText = isGroupMute ? mute_text : (isGroupShow ? show_text : hide_text) | |
| let pressTimer | |
| let firstTime =0, lastTime =0 | |
| let isHolding = false | |
| see_btn.addEventListener('click', () => { | |
| if(isHolding){ | |
| isHolding = false | |
| return | |
| } | |
| for (const node of nodesInGroup) { | |
| node.mode = isGroupShow ? 4 : 0; | |
| node.graph.change(); | |
| } | |
| isGroupShow = nodesInGroup[0].mode == 0 ? true : false | |
| isGroupMute = nodesInGroup[0].mode == 2 ? true : false | |
| see_btn.style = isGroupMute ? `background-color:var(--error-text);color:var(--input-text);` + defaultStyle : (isGroupShow ? `background-color:#006691;color:var(--input-text);` + defaultStyle : `background-color: var(--comfy-input-bg);color:var(--descrip-text);` + defaultStyle) | |
| see_btn.innerText = isGroupMute ? mute_text : (isGroupShow ? show_text : hide_text) | |
| }) | |
| see_btn.addEventListener('mousedown', () => { | |
| firstTime = new Date().getTime(); | |
| clearTimeout(pressTimer); | |
| pressTimer = setTimeout(_=>{ | |
| for (const node of nodesInGroup) { | |
| node.mode = isGroupMute ? 0 : 2; | |
| node.graph.change(); | |
| } | |
| isGroupShow = nodesInGroup[0].mode == 0 ? true : false | |
| isGroupMute = nodesInGroup[0].mode == 2 ? true : false | |
| see_btn.style = isGroupMute ? `background-color:var(--error-text);color:var(--input-text);` + defaultStyle : (isGroupShow ? `background-color:#006691;color:var(--input-text);` + defaultStyle : `background-color: var(--comfy-input-bg);color:var(--descrip-text);` + defaultStyle) | |
| see_btn.innerText = isGroupMute ? mute_text : (isGroupShow ? show_text : hide_text) | |
| },500) | |
| }) | |
| see_btn.addEventListener('mouseup', () => { | |
| lastTime = new Date().getTime(); | |
| if(lastTime - firstTime > 500) isHolding = true | |
| clearTimeout(pressTimer); | |
| }) | |
| buttons.append(see_btn) | |
| group_item.append(buttons) | |
| groupsDiv.append(group_item) | |
| } | |
| } | |
| let groupsDiv = document.createElement('div') | |
| groupsDiv.id = 'easyuse-groups-items' | |
| groupsDiv.style = `overflow-y: auto;max-height: 400px;height:100%;width: 100%;` | |
| let autoSortDiv = document.createElement('button') | |
| autoSortDiv.style = `cursor:pointer;font-size:10px;padding:2px 4px;color:var(--input-text);background-color: var(--comfy-input-bg);border: 1px solid var(--border-color);border-radius:4px;` | |
| autoSortDiv.innerText = $t('Auto Sorting') | |
| autoSortDiv.addEventListener('click',e=>{ | |
| e.preventDefault() | |
| groupsDiv.innerHTML = `` | |
| let new_groups = groups.sort((a,b)=> a['pos'][0] - b['pos'][0]).sort((a,b)=> a['pos'][1] - b['pos'][1]) | |
| updateGroups(new_groups, groupsDiv, autoSortDiv) | |
| }) | |
| updateGroups(groups, groupsDiv, autoSortDiv) | |
| div.appendChild(groupsDiv) | |
| let remarkDiv = document.createElement('p') | |
| remarkDiv.style = `text-align:center; font-size:10px; padding:0 10px;color:var(--descrip-text)` | |
| remarkDiv.innerText = $t('Toggle `Show/Hide` can set mode of group, LongPress can set group nodes to never') | |
| div.appendChild(groupsDiv) | |
| div.appendChild(remarkDiv) | |
| div.appendChild(autoSortDiv) | |
| let graphDiv = document.getElementById("graph-canvas") | |
| graphDiv.addEventListener('mouseover', async () => { | |
| groupsDiv.innerHTML = `` | |
| let new_groups = app.canvas.graph._groups | |
| updateGroups(new_groups, groupsDiv, autoSortDiv) | |
| old_nodes = nodes | |
| }) | |
| if (!document.querySelector('#easyuse_groups_map')){ | |
| document.body.appendChild(div) | |
| }else{ | |
| div.style.display = 'flex' | |
| } | |
| } | |
| async function cleanup(){ | |
| try { | |
| const {Running, Pending} = await api.getQueue() | |
| if(Running.length>0 || Pending.length>0){ | |
| toast.error($t("Clean Failed")+ ":"+ $t("Please stop all running tasks before cleaning GPU")) | |
| return | |
| } | |
| api.fetchApi("/easyuse/cleangpu",{ | |
| method:"POST" | |
| }).then(res=>{ | |
| if(res.status == 200){ | |
| toast.success($t("Clean SuccessFully")) | |
| }else{ | |
| toast.error($t("Clean Failed")) | |
| } | |
| }) | |
| } catch (exception) {} | |
| } | |
| let guideDialog = null | |
| let isDownloading = false | |
| function download_model(url,local_dir){ | |
| if(isDownloading || !url || !local_dir) return | |
| isDownloading = true | |
| let body = new FormData(); | |
| body.append('url', url); | |
| body.append('local_dir', local_dir); | |
| api.fetchApi("/easyuse/model/download",{ | |
| method:"POST", | |
| body | |
| }).then(res=>{ | |
| if(res.status == 200){ | |
| toast.success($t("Download SuccessFully")) | |
| }else{ | |
| toast.error($t("Download Failed")) | |
| } | |
| isDownloading = false | |
| }) | |
| } | |
| class GuideDialog { | |
| constructor(note, need_models){ | |
| this.dialogDiv = null | |
| this.modelsDiv = null | |
| if(need_models?.length>0){ | |
| let tbody = [] | |
| for(let i=0;i<need_models.length;i++){ | |
| tbody.push($el('tr',[ | |
| $el('td',{innerHTML:need_models[i].title || need_models[i].name || ''}), | |
| $el('td',[ | |
| need_models[i]['download_url'] ? $el('a',{onclick:_=>download_model(need_models[i]['download_url'],need_models[i]['local_dir']), target:"_blank", textContent:$t('Download Model')}) : '', | |
| need_models[i]['source_url'] ? $el('a',{href:need_models[i]['source_url'], target:"_blank", textContent:$t('Source Url')}) : '', | |
| need_models[i]['desciption'] ? $el('span',{textContent:need_models[i]['desciption']}) : '', | |
| ]), | |
| ])) | |
| } | |
| this.modelsDiv = $el('div.easyuse-guide-dialog-models.markdown-body',[ | |
| $el('h3',{textContent:$t('Models Required')}), | |
| $el('table',{cellpadding:0,cellspacing:0},[ | |
| $el('thead',[ | |
| $el('tr',[ | |
| $el('th',{innerHTML:$t('ModelName')}), | |
| $el('th',{innerHTML:$t('Description')}), | |
| ]) | |
| ]), | |
| $el('tbody',tbody) | |
| ]) | |
| ]) | |
| } | |
| this.dialogDiv = $el('div.easyuse-guide-dialog.hidden',[ | |
| $el('div.easyuse-guide-dialog-header',[ | |
| $el('div.easyuse-guide-dialog-top',[ | |
| $el('div.easyuse-guide-dialog-title',{ | |
| innerHTML:$t('Workflow Guide') | |
| }), | |
| $el('button.closeBtn',{innerHTML:closeIcon,onclick:_=>this.close()}) | |
| ]), | |
| $el('div.easyuse-guide-dialog-remark',{ | |
| innerHTML:`${$t('Workflow created by')} <a href="https://github.com/yolain/" target="_blank">Yolain</a> , ${$t('Watch more video content')} <a href="https://space.bilibili.com/1840885116" target="_blank">B站乱乱呀</a>` | |
| }) | |
| ]), | |
| $el('div.easyuse-guide-dialog-content.markdown-body',[ | |
| $el('div.easyuse-guide-dialog-note',{ | |
| innerHTML:note | |
| }), | |
| ...this.modelsDiv ? [this.modelsDiv] : [] | |
| ]) | |
| ]) | |
| if(disableRenderInfo){ | |
| this.dialogDiv.classList.add('disable-render-info') | |
| } | |
| document.body.appendChild(this.dialogDiv) | |
| } | |
| show(){ | |
| if(this.dialogDiv) this.dialogDiv.classList.remove('hidden') | |
| } | |
| close(){ | |
| if(this.dialogDiv){ | |
| this.dialogDiv.classList.add('hidden') | |
| } | |
| } | |
| toggle(){ | |
| if(this.dialogDiv){ | |
| if(this.dialogDiv.classList.contains('hidden')){ | |
| this.show() | |
| }else{ | |
| this.close() | |
| } | |
| } | |
| } | |
| remove(){ | |
| if(this.dialogDiv) document.body.removeChild(this.dialogDiv) | |
| } | |
| } | |
| // toolbar | |
| const toolBarId = "Comfy.EasyUse.toolBar" | |
| const getEnableToolBar = _ => app.ui.settings.getSettingValue(toolBarId, true) | |
| const getNewMenuPosition = _ => { | |
| try{ | |
| return app.ui.settings.getSettingValue('Comfy.UseNewMenu', 'Disabled') | |
| }catch (e){ | |
| return 'Disabled' | |
| } | |
| } | |
| let note = null | |
| let toolbar = null | |
| let enableToolBar = getEnableToolBar() && getNewMenuPosition() == 'Disabled' | |
| let disableRenderInfo = localStorage['Comfy.Settings.Comfy.EasyUse.disableRenderInfo'] ? true : false | |
| export function addToolBar(app) { | |
| app.ui.settings.addSetting({ | |
| id: toolBarId, | |
| name: $t("Enable tool bar fixed on the left-bottom (ComfyUI-Easy-Use)"), | |
| type: "boolean", | |
| defaultValue: enableToolBar, | |
| onChange(value) { | |
| enableToolBar = !!value; | |
| if(enableToolBar){ | |
| showToolBar() | |
| }else hideToolBar() | |
| }, | |
| }); | |
| } | |
| function showToolBar(){ | |
| if(toolbar) toolbar.style.display = 'flex' | |
| } | |
| function hideToolBar(){ | |
| if(toolbar) toolbar.style.display = 'none' | |
| } | |
| let monitor = null | |
| function setCrystoolsUI(position){ | |
| const crystools = document.getElementById('crystools-root')?.children || null | |
| if(crystools?.length>0){ | |
| if(!monitor){ | |
| for (let i = 0; i < crystools.length; i++) { | |
| if (crystools[i].id === 'crystools-monitor-container') { | |
| monitor = crystools[i]; | |
| break; | |
| } | |
| } | |
| } | |
| if(monitor){ | |
| if(position == 'Disabled'){ | |
| let replace = true | |
| for (let i = 0; i < crystools.length; i++) { | |
| if (crystools[i].id === 'crystools-monitor-container') { | |
| replace = false | |
| break; | |
| } | |
| } | |
| document.getElementById('crystools-root').appendChild(monitor) | |
| } | |
| else { | |
| let monitor_div = document.getElementById('comfyui-menu-monitor') | |
| if(!monitor_div) app.menu.settingsGroup.element.before($el('div',{id:'comfyui-menu-monitor'},monitor)) | |
| else monitor_div.appendChild(monitor) | |
| } | |
| } | |
| } | |
| } | |
| const changeNewMenuPosition = app.ui.settings.settingsLookup?.['Comfy.UseNewMenu'] | |
| if(changeNewMenuPosition) changeNewMenuPosition.onChange = v => { | |
| v == 'Disabled' ? showToolBar() : hideToolBar() | |
| setCrystoolsUI(v) | |
| } | |
| app.registerExtension({ | |
| name: "comfy.easyUse", | |
| init() { | |
| // Canvas Menu | |
| const getCanvasMenuOptions = LGraphCanvas.prototype.getCanvasMenuOptions; | |
| LGraphCanvas.prototype.getCanvasMenuOptions = function () { | |
| const options = getCanvasMenuOptions.apply(this, arguments); | |
| let emptyImg = new Image() | |
| emptyImg.src = ""; | |
| options.push(null, | |
| // Groups Map | |
| { | |
| content: groupIcon.replace('currentColor','var(--warning-color)') + ' '+ $t('Groups Map') + ' (EasyUse)', | |
| callback: async() => { | |
| createGroupMap() | |
| } | |
| }, | |
| // Force clean ComfyUI GPU Used 强制卸载模型GPU占用 | |
| { | |
| content: rocketIcon.replace('currentColor','var(--theme-color-light)') + ' '+ $t('Cleanup Of GPU Usage') + ' (EasyUse)', | |
| callback: async() =>{ | |
| await cleanup() | |
| } | |
| }, | |
| // Only show the reboot option if the server is running on a local network 仅在本地或局域网环境可重启服务 | |
| isLocalNetwork(window.location.host) ? { | |
| content: rebootIcon.replace('currentColor','var(--error-color)') + ' '+ $t('Reboot ComfyUI') + ' (EasyUse)', | |
| callback: _ =>{ | |
| if (confirm($t("Are you sure you'd like to reboot the server?"))){ | |
| try { | |
| api.fetchApi("/easyuse/reboot"); | |
| } catch (exception) {} | |
| } | |
| } | |
| } : null, | |
| ); | |
| return options; | |
| }; | |
| let renderInfoEvent = LGraphCanvas.prototype.renderInfo | |
| if(disableRenderInfo){ | |
| LGraphCanvas.prototype.renderInfo = function (ctx, x, y) {} | |
| } | |
| if(!toolbar){ | |
| toolbar = $el('div.easyuse-toolbar',[ | |
| $el('div.easyuse-toolbar-item',{ | |
| onclick:_=>{ | |
| createGroupMap() | |
| } | |
| },[ | |
| $el('div.easyuse-toolbar-icon.group', {innerHTML:groupIcon}), | |
| $el('div.easyuse-toolbar-tips',$t('Groups Map')) | |
| ]), | |
| $el('div.easyuse-toolbar-item',{ | |
| onclick:async()=>{ | |
| await cleanup() | |
| } | |
| },[ | |
| $el('div.easyuse-toolbar-icon.rocket',{innerHTML:rocketIcon}), | |
| $el('div.easyuse-toolbar-tips',$t('Cleanup Of GPU Usage')) | |
| ]), | |
| ]) | |
| if(disableRenderInfo){ | |
| toolbar.classList.add('disable-render-info') | |
| }else{ | |
| toolbar.classList.remove('disable-render-info') | |
| } | |
| document.body.appendChild(toolbar) | |
| } | |
| // rewrite handleFile | |
| let loadGraphDataEvent = app.loadGraphData | |
| app.loadGraphData = async function (data, clean=true) { | |
| // if(data?.extra?.cpr){ | |
| // toast.copyright() | |
| // } | |
| if(data?.extra?.note){ | |
| if(guideDialog) { | |
| guideDialog.remove() | |
| guideDialog = null | |
| } | |
| if(note && toolbar) toolbar.removeChild(note) | |
| const need_models = data.extra?.need_models || null | |
| guideDialog = new GuideDialog(data.extra.note, need_models) | |
| note = $el('div.easyuse-toolbar-item',{ | |
| onclick:async()=>{ | |
| guideDialog.toggle() | |
| } | |
| },[ | |
| $el('div.easyuse-toolbar-icon.question',{innerHTML:quesitonIcon}), | |
| $el('div.easyuse-toolbar-tips',$t('Workflow Guide')) | |
| ]) | |
| if(toolbar) toolbar.insertBefore(note, toolbar.firstChild) | |
| } | |
| else{ | |
| if(note) { | |
| toolbar.removeChild(note) | |
| note = null | |
| } | |
| } | |
| return await loadGraphDataEvent.apply(this, [...arguments]) | |
| } | |
| addToolBar(app) | |
| }, | |
| async setup() { | |
| // New style menu button | |
| if(app.menu?.actionsGroup){ | |
| const groupMap = new (await import('../../../../scripts/ui/components/button.js')).ComfyButton({ | |
| icon:'list-box', | |
| action:()=> createGroupMap(), | |
| tooltip: "EasyUse Group Map", | |
| // content: "EasyUse Group Map", | |
| classList: "comfyui-button comfyui-menu-mobile-collapse" | |
| }); | |
| app.menu.actionsGroup.element.after(groupMap.element); | |
| const position = getNewMenuPosition() | |
| setCrystoolsUI(position) | |
| if(position == 'Disabled') showToolBar() | |
| else hideToolBar() | |
| // const easyNewMenu = $el('div.easyuse-new-menu',[ | |
| // $el('div.easyuse-new-menu-intro',[ | |
| // $el('div.easyuse-new-menu-logo',{innerHTML:logoIcon}), | |
| // $el('div.easyuse-new-menu-title',[ | |
| // $el('div.title',{textContent:'ComfyUI-Easy-Use'}), | |
| // $el('div.desc',{textContent:'Version:'}) | |
| // ]) | |
| // ]) | |
| // ]) | |
| // app.menu?.actionsGroup.element.after(new (await import('../../../../scripts/ui/components/splitButton.js')).ComfySplitButton({ | |
| // primary: groupMap, | |
| // mode:'click', | |
| // position:'absolute', | |
| // horizontal: 'right' | |
| // },easyNewMenu).element); | |
| } | |
| }, | |
| beforeRegisterNodeDef(nodeType, nodeData, app) { | |
| if (nodeData.name.startsWith("easy")) { | |
| const origOnConfigure = nodeType.prototype.onConfigure; | |
| nodeType.prototype.onConfigure = function () { | |
| const r = origOnConfigure ? origOnConfigure.apply(this, arguments) : undefined; | |
| return r; | |
| }; | |
| } | |
| }, | |
| }); |