const contextMenuInit = () => { let eventListenerApplied = false; const menuSpecs = new Map(); const uid = () => Date.now().toString(36) + Math.random().toString(36).substring(2); function showContextMenu(event, element, menuEntries) { const posx = event.clientX + document.body.scrollLeft + document.documentElement.scrollLeft; const posy = event.clientY + document.body.scrollTop + document.documentElement.scrollTop; const oldMenu = gradioApp().querySelector('#context-menu'); if (oldMenu) oldMenu.remove(); const contextMenu = document.createElement('nav'); contextMenu.id = 'context-menu'; contextMenu.style.top = `${posy}px`; contextMenu.style.left = `${posx}px`; const contextMenuList = document.createElement('ul'); contextMenuList.className = 'context-menu-items'; contextMenu.append(contextMenuList); menuEntries.forEach((entry) => { const contextMenuEntry = document.createElement('a'); contextMenuEntry.innerHTML = entry.name; contextMenuEntry.addEventListener('click', (e) => entry.func()); contextMenuList.append(contextMenuEntry); }); gradioApp().appendChild(contextMenu); const menuWidth = contextMenu.offsetWidth + 4; const menuHeight = contextMenu.offsetHeight + 4; const windowWidth = window.innerWidth; const windowHeight = window.innerHeight; if ((windowWidth - posx) < menuWidth) contextMenu.style.left = `${windowWidth - menuWidth}px`; if ((windowHeight - posy) < menuHeight) contextMenu.style.top = `${windowHeight - menuHeight}px`; } function appendContextMenuOption(targetElementSelector, entryName, entryFunction) { let currentItems = menuSpecs.get(targetElementSelector); if (!currentItems) { currentItems = []; menuSpecs.set(targetElementSelector, currentItems); } const newItem = { id: `${targetElementSelector}_${uid()}`, name: entryName, func: entryFunction, isNew: true, }; currentItems.push(newItem); return newItem.id; } function removeContextMenuOption(id) { menuSpecs.forEach((v, k) => { let index = -1; v.forEach((e, ei) => { if (e.id === id) { index = ei; } }); if (index >= 0) v.splice(index, 1); }); } async function addContextMenuEventListener() { if (eventListenerApplied) return; log('initContextMenu'); gradioApp().addEventListener('click', (e) => { if (!e.isTrusted) return; const oldMenu = gradioApp().querySelector('#context-menu'); if (oldMenu) oldMenu.remove(); }); gradioApp().addEventListener('contextmenu', (e) => { const oldMenu = gradioApp().querySelector('#context-menu'); if (oldMenu) oldMenu.remove(); menuSpecs.forEach((v, k) => { if (e.composedPath()[0].matches(k)) { showContextMenu(e, e.composedPath()[0], v); e.preventDefault(); } }); }); eventListenerApplied = true; } return [appendContextMenuOption, removeContextMenuOption, addContextMenuEventListener]; }; const initResponse = contextMenuInit(); const appendContextMenuOption = initResponse[0]; const removeContextMenuOption = initResponse[1]; const addContextMenuEventListener = initResponse[2]; async function initContextMenu() { const generateForever = (genbuttonid, interruptbuttonid) => { if (window.generateOnRepeatInterval) { log('generateForever: cancel'); clearInterval(window.generateOnRepeatInterval); window.generateOnRepeatInterval = null; } else { log('generateForever: start'); const genbutton = gradioApp().querySelector(genbuttonid); const busy = document.getElementById('progressbar')?.style.display === 'block'; if (!busy) genbutton.click(); window.generateOnRepeatInterval = setInterval(() => { const pbBusy = document.getElementById('progressbar')?.style.display === 'block'; if (!pbBusy) genbutton.click(); }, 500); } }; for (const tab of ['txt2img', 'img2img', 'control']) { for (const el of ['generate', 'interrupt', 'skip', 'pause', 'paste', 'clear_prompt', 'extra_networks_btn']) { const id = `#${tab}_${el}`; appendContextMenuOption(id, 'Copy to clipboard', () => navigator.clipboard.writeText(document.querySelector(`#${tab}_prompt > label > textarea`).value)); appendContextMenuOption(id, 'Generate forever', () => generateForever(`#${tab}_generate`)); appendContextMenuOption(id, 'Apply selected style', quickApplyStyle); appendContextMenuOption(id, 'Quick save style', quickSaveStyle); appendContextMenuOption(id, 'nVidia overlay', initNVML); } } } onUiLoaded(initContextMenu); onAfterUiUpdate(() => addContextMenuEventListener());