Some checks failed
Python Linting / Run Ruff (push) Has been cancelled
Python Linting / Run Pylint (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.10, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.11, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.12, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-unix-nightly (12.1, , linux, 3.11, [self-hosted Linux], nightly) (push) Has been cancelled
Execution Tests / test (macos-latest) (push) Has been cancelled
Execution Tests / test (ubuntu-latest) (push) Has been cancelled
Execution Tests / test (windows-latest) (push) Has been cancelled
Test server launches without errors / test (push) Has been cancelled
Unit Tests / test (macos-latest) (push) Has been cancelled
Unit Tests / test (ubuntu-latest) (push) Has been cancelled
Unit Tests / test (windows-2022) (push) Has been cancelled
Includes 30 custom nodes committed directly, 7 Civitai-exclusive loras stored via Git LFS, and a setup script that installs all dependencies and downloads HuggingFace-hosted models on vast.ai. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
610 lines
24 KiB
JavaScript
610 lines
24 KiB
JavaScript
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 = "data:image/gif;base64,R0lGODlhAQABAIAAAAUEBAAAACwAAAAAAQABAAACAkQBADs=";
|
|
|
|
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;
|
|
};
|
|
}
|
|
},
|
|
}); |