|
|
import { app } from '../../../scripts/app.js' |
|
|
import { api } from '../../../scripts/api.js' |
|
|
import { setWidgetConfig } from '../../../extensions/core/widgetInputs.js' |
|
|
import { applyTextReplacements } from "../../../scripts/utils.js"; |
|
|
|
|
|
function chainCallback(object, property, callback) { |
|
|
if (object == undefined) { |
|
|
|
|
|
console.error("Tried to add callback to non-existant object") |
|
|
return; |
|
|
} |
|
|
if (property in object && object[property]) { |
|
|
const callback_orig = object[property] |
|
|
object[property] = function () { |
|
|
const r = callback_orig.apply(this, arguments); |
|
|
return callback.apply(this, arguments) ?? r |
|
|
}; |
|
|
} else { |
|
|
object[property] = callback; |
|
|
} |
|
|
} |
|
|
|
|
|
function getNodeById(id, graph=app.graph) { |
|
|
let cg = graph |
|
|
let node = undefined |
|
|
for (let sid of (''+id).split(':')) { |
|
|
node = cg?.getNodeById?.(sid) |
|
|
cg = node?.subgraph |
|
|
} |
|
|
return node |
|
|
} |
|
|
|
|
|
const convDict = { |
|
|
VHS_LoadImages : ["directory", null, "image_load_cap", "skip_first_images", "select_every_nth"], |
|
|
VHS_LoadImagesPath : ["directory", "image_load_cap", "skip_first_images", "select_every_nth"], |
|
|
VHS_VideoCombine : ["frame_rate", "loop_count", "filename_prefix", "format", "pingpong", "save_image"], |
|
|
VHS_LoadVideo : ["video", "force_rate", "force_size", "frame_load_cap", "skip_first_frames", "select_every_nth"], |
|
|
VHS_LoadVideoPath : ["video", "force_rate", "force_size", "frame_load_cap", "skip_first_frames", "select_every_nth"], |
|
|
}; |
|
|
const renameDict = {VHS_VideoCombine : {save_output : "save_image"}} |
|
|
function useKVState(nodeType) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function () { |
|
|
chainCallback(this, "onConfigure", function(info) { |
|
|
if (!this.widgets) { |
|
|
|
|
|
return |
|
|
} |
|
|
if (typeof(info.widgets_values) != "object") { |
|
|
|
|
|
return |
|
|
} |
|
|
let widgetDict = info.widgets_values |
|
|
if (info.widgets_values.length) { |
|
|
|
|
|
if (this.type in convDict) { |
|
|
|
|
|
let convList = convDict[this.type]; |
|
|
if(info.widgets_values.length >= convList.length) { |
|
|
|
|
|
widgetDict = {} |
|
|
for (let i = 0; i < convList.length; i++) { |
|
|
if(!convList[i]) { |
|
|
|
|
|
continue |
|
|
} |
|
|
widgetDict[convList[i]] = info.widgets_values[i]; |
|
|
} |
|
|
} else { |
|
|
|
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
if ('force_size' in widgetDict) { |
|
|
|
|
|
if (widgetDict.force_size.includes?.('x')) { |
|
|
let sizes = widgetDict.force_size.split('x') |
|
|
if (sizes[0] != '?') { |
|
|
widgetDict.custom_width = parseInt(sizes[0]) |
|
|
} else { |
|
|
widgetDict.custom_width = 0 |
|
|
} |
|
|
if (sizes[1] != '?') { |
|
|
widgetDict.custom_height = parseInt(sizes[1]) |
|
|
} else { |
|
|
widgetDict.custom_height = 0 |
|
|
} |
|
|
} else { |
|
|
if (['Disabled', 'Custom Height'].includes(widgetDict.force_size)) { |
|
|
widgetDict.custom_width = 0 |
|
|
} |
|
|
if (['Disabled', 'Custom Width'].includes(widgetDict.force_size)) { |
|
|
widgetDict.custom_height = 0 |
|
|
} |
|
|
} |
|
|
} |
|
|
if (widgetDict.videopreview?.params?.force_size) { |
|
|
delete widgetDict.videopreview.params.force_size |
|
|
} |
|
|
let inputs = {} |
|
|
for (let i of this.inputs) { |
|
|
inputs[i.name] = i |
|
|
} |
|
|
if (widgetDict.length == undefined) { |
|
|
for (let w of this.widgets) { |
|
|
if (w.type =="button") { |
|
|
continue |
|
|
} |
|
|
if (w.name in widgetDict) { |
|
|
w.value = widgetDict[w.name]; |
|
|
w.callback?.(w.value) |
|
|
} else { |
|
|
|
|
|
if (this.type in renameDict && w.name in renameDict[this.type]) { |
|
|
if (renameDict[this.type][w.name] in widgetDict) { |
|
|
w.value = widgetDict[renameDict[this.type][w.name]] |
|
|
w.callback?.(w.value) |
|
|
continue |
|
|
} |
|
|
} |
|
|
|
|
|
let inputs = LiteGraph.getNodeType(this.type).nodeData.input; |
|
|
let initialValue = null; |
|
|
if (inputs?.required?.hasOwnProperty(w.name)) { |
|
|
if (inputs.required[w.name][1]?.hasOwnProperty("default")) { |
|
|
initialValue = inputs.required[w.name][1].default; |
|
|
} else if (inputs.required[w.name][0].length) { |
|
|
initialValue = inputs.required[w.name][0][0]; |
|
|
} |
|
|
} else if (inputs?.optional?.hasOwnProperty(w.name)) { |
|
|
if (inputs.optional[w.name][1]?.hasOwnProperty("default")) { |
|
|
initialValue = inputs.optional[w.name][1].default; |
|
|
} else if (inputs.optional[w.name][0].length) { |
|
|
initialValue = inputs.optional[w.name][0][0]; |
|
|
} |
|
|
} |
|
|
if (initialValue) { |
|
|
w.value = initialValue; |
|
|
w.callback?.(w.value) |
|
|
} |
|
|
} |
|
|
if (w.name in inputs && w.config) { |
|
|
setWidgetConfig(inputs[w.name], w.config) |
|
|
} |
|
|
} |
|
|
} else { |
|
|
|
|
|
|
|
|
|
|
|
if (info?.widgets_values?.length != this.widgets.length) { |
|
|
|
|
|
|
|
|
app.ui.dialog.show("Failed to restore node: " + this.title + "\nPlease remove and re-add it.") |
|
|
this.bgcolor = "#C00" |
|
|
} |
|
|
} |
|
|
}); |
|
|
chainCallback(this, "onSerialize", function(info) { |
|
|
info.widgets_values = {}; |
|
|
if (!this.widgets) { |
|
|
|
|
|
return; |
|
|
} |
|
|
for (let w of this.widgets) { |
|
|
info.widgets_values[w.name] = w.value; |
|
|
} |
|
|
}); |
|
|
}) |
|
|
} |
|
|
var helpDOM; |
|
|
if (!app.helpDOM) { |
|
|
helpDOM = document.createElement("div"); |
|
|
app.VHSHelp = helpDOM |
|
|
} |
|
|
function initHelpDOM() { |
|
|
let parentDOM = document.createElement("div"); |
|
|
parentDOM.className = "VHS_floatinghelp" |
|
|
document.body.appendChild(parentDOM) |
|
|
parentDOM.appendChild(helpDOM) |
|
|
helpDOM.className = "litegraph"; |
|
|
let scrollbarStyle = document.createElement('style'); |
|
|
scrollbarStyle.innerHTML = ` |
|
|
.VHS_floatinghelp { |
|
|
scrollbar-width: 6px; |
|
|
scrollbar-color: #0003 #0000; |
|
|
&::-webkit-scrollbar { |
|
|
background: transparent; |
|
|
width: 6px; |
|
|
} |
|
|
&::-webkit-scrollbar-thumb { |
|
|
background: #0005; |
|
|
border-radius: 20px |
|
|
} |
|
|
&::-webkit-scrollbar-button { |
|
|
display: none; |
|
|
} |
|
|
} |
|
|
.VHS_loopedvideo::-webkit-media-controls-mute-button { |
|
|
display:none; |
|
|
} |
|
|
.VHS_loopedvideo::-webkit-media-controls-fullscreen-button { |
|
|
display:none; |
|
|
} |
|
|
` |
|
|
scrollbarStyle.id = 'scroll-properties' |
|
|
parentDOM.appendChild(scrollbarStyle) |
|
|
chainCallback(app.canvas, "onDrawForeground", function (ctx, visible_rect){ |
|
|
let n = helpDOM.node |
|
|
if (!n || !n?.graph) { |
|
|
parentDOM.style['left'] = '-5000px' |
|
|
return |
|
|
} |
|
|
|
|
|
|
|
|
const transform = ctx.getTransform(); |
|
|
const scale = app.canvas.ds.scale; |
|
|
|
|
|
const bcr = app.canvas.canvas.getBoundingClientRect() |
|
|
const x = transform.e*scale/transform.a + bcr.x; |
|
|
const y = transform.f*scale/transform.a + bcr.y; |
|
|
|
|
|
Object.assign(parentDOM.style, { |
|
|
left: (x+(n.pos[0] + n.size[0]+15)*scale) + "px", |
|
|
top: (y+(n.pos[1]-LiteGraph.NODE_TITLE_HEIGHT)*scale) + "px", |
|
|
width: "400px", |
|
|
minHeight: "100px", |
|
|
maxHeight: "600px", |
|
|
overflowY: 'scroll', |
|
|
transformOrigin: '0 0', |
|
|
transform: 'scale(' + scale + ',' + scale +')', |
|
|
fontSize: '18px', |
|
|
backgroundColor: LiteGraph.NODE_DEFAULT_BGCOLOR, |
|
|
boxShadow: '0 0 10px black', |
|
|
borderRadius: '4px', |
|
|
padding: '3px', |
|
|
zIndex: 3, |
|
|
position: "absolute", |
|
|
display: 'inline', |
|
|
}); |
|
|
}); |
|
|
function setCollapse(el, doCollapse) { |
|
|
if (doCollapse) { |
|
|
el.children[0].children[0].innerHTML = '+' |
|
|
Object.assign(el.children[1].style, { |
|
|
color: '#CCC', |
|
|
overflowX: 'hidden', |
|
|
width: '0px', |
|
|
minWidth: 'calc(100% - 20px)', |
|
|
textOverflow: 'ellipsis', |
|
|
whiteSpace: 'nowrap', |
|
|
}) |
|
|
for (let child of el.children[1].children) { |
|
|
if (child.style.display != 'none'){ |
|
|
child.origDisplay = child.style.display |
|
|
} |
|
|
child.style.display = 'none' |
|
|
} |
|
|
} else { |
|
|
el.children[0].children[0].innerHTML = '-' |
|
|
Object.assign(el.children[1].style, { |
|
|
color: '', |
|
|
overflowX: '', |
|
|
width: '100%', |
|
|
minWidth: '', |
|
|
textOverflow: '', |
|
|
whiteSpace: '', |
|
|
}) |
|
|
for (let child of el.children[1].children) { |
|
|
child.style.display = child.origDisplay |
|
|
} |
|
|
} |
|
|
} |
|
|
helpDOM.collapseOnClick = function() { |
|
|
let doCollapse = this.children[0].innerHTML == '-' |
|
|
setCollapse(this.parentElement, doCollapse) |
|
|
} |
|
|
helpDOM.selectHelp = function(name, value) { |
|
|
|
|
|
function collapseUnlessMatch(items,t) { |
|
|
var match = items.querySelector('[vhs_title="' + t + '"]') |
|
|
if (!match) { |
|
|
for (let i of items.children) { |
|
|
if (i.innerHTML.slice(0,t.length+5).includes(t)) { |
|
|
match = i |
|
|
break |
|
|
} |
|
|
} |
|
|
} |
|
|
if (!match) { |
|
|
return null |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
match.scrollIntoView(false) |
|
|
window.scrollTo(0,0) |
|
|
for (let i of items.querySelectorAll('.VHS_collapse')) { |
|
|
if (i.contains(match)) { |
|
|
setCollapse(i, false) |
|
|
} else { |
|
|
setCollapse(i, true) |
|
|
} |
|
|
} |
|
|
return match |
|
|
} |
|
|
let target = collapseUnlessMatch(helpDOM, name) |
|
|
if (target && value) { |
|
|
collapseUnlessMatch(target, value) |
|
|
} |
|
|
} |
|
|
let titleContext = document.createElement("canvas").getContext("2d") |
|
|
titleContext.font = app.canvas.title_text_font; |
|
|
helpDOM.calculateTitleLength = function(text) { |
|
|
return titleContext.measureText(text).width |
|
|
} |
|
|
helpDOM.addHelp = function(node, nodeType, description) { |
|
|
if (!description) { |
|
|
return |
|
|
} |
|
|
|
|
|
let originalComputeSize = node.computeSize |
|
|
node.computeSize = function() { |
|
|
let size = originalComputeSize.apply(this, arguments) |
|
|
if (!this.title) { |
|
|
return size |
|
|
} |
|
|
let title_width = helpDOM.calculateTitleLength(this.title) |
|
|
size[0] = Math.max(size[0], title_width + LiteGraph.NODE_TITLE_HEIGHT*2) |
|
|
return size |
|
|
} |
|
|
|
|
|
node.description = description |
|
|
chainCallback(node, "onDrawForeground", function (ctx) { |
|
|
if (this?.flags?.collapsed) { |
|
|
return |
|
|
} |
|
|
|
|
|
ctx.save() |
|
|
ctx.font = 'bold 20px Arial' |
|
|
ctx.fillText("?", this.size[0]-17, -8) |
|
|
ctx.restore() |
|
|
}) |
|
|
chainCallback(node, "onMouseDown", function (e, pos, canvas) { |
|
|
if (this?.flags?.collapsed) { |
|
|
return |
|
|
} |
|
|
|
|
|
if (pos[1] < 0 && pos[0] + LiteGraph.NODE_TITLE_HEIGHT > this.size[0]) { |
|
|
|
|
|
if (helpDOM.node == this) { |
|
|
helpDOM.node = undefined |
|
|
} else { |
|
|
helpDOM.node = this; |
|
|
helpDOM.innerHTML = this.description || "no help provided " |
|
|
for (let e of helpDOM.querySelectorAll('.VHS_collapse')) { |
|
|
e.children[0].onclick = helpDOM.collapseOnClick |
|
|
e.children[0].style.cursor = 'pointer' |
|
|
} |
|
|
for (let e of helpDOM.querySelectorAll('.VHS_precollapse')) { |
|
|
setCollapse(e, true) |
|
|
} |
|
|
for (let e of helpDOM.querySelectorAll('.VHS_loopedvideo')) { |
|
|
e?.play() |
|
|
} |
|
|
helpDOM.parentElement.scrollTo(0,0) |
|
|
} |
|
|
return true |
|
|
} |
|
|
}) |
|
|
let timeout = null |
|
|
chainCallback(node, "onMouseMove", function (e, pos, canvas) { |
|
|
if (timeout) { |
|
|
clearTimeout(timeout) |
|
|
timeout = null |
|
|
} |
|
|
if (helpDOM.node != this) { |
|
|
return |
|
|
} |
|
|
timeout = setTimeout(() => { |
|
|
let n = this |
|
|
if (pos[0] > 0 && pos[0] < n.size[0] |
|
|
&& pos[1] > 0 && pos[1] < n.size[1]) { |
|
|
|
|
|
let inputRows = Math.max(n.inputs?.length || 0, n.outputs?.length || 0) |
|
|
if (pos[1] < LiteGraph.NODE_SLOT_HEIGHT * inputRows) { |
|
|
let row = Math.floor((pos[1] - 7) / LiteGraph.NODE_SLOT_HEIGHT) |
|
|
if (pos[0] < n.size[0]/2) { |
|
|
if (row < n.inputs.length) { |
|
|
helpDOM.selectHelp(n.inputs[row].name) |
|
|
} |
|
|
} else { |
|
|
if (row < n.outputs.length) { |
|
|
helpDOM.selectHelp(n.outputs[row].name) |
|
|
} |
|
|
} |
|
|
} else { |
|
|
|
|
|
let basey = LiteGraph.NODE_SLOT_HEIGHT * inputRows + 6 |
|
|
for (let w of n.widgets) { |
|
|
if (w.y) { |
|
|
basey = w.y |
|
|
} |
|
|
let wheight = LiteGraph.NODE_WIDGET_HEIGHT+4 |
|
|
if (w.computeSize) { |
|
|
wheight = w.computeSize(n.size[0])[1] |
|
|
} |
|
|
if (pos[1] < basey + wheight) { |
|
|
helpDOM.selectHelp(w.name, w.value) |
|
|
break |
|
|
} |
|
|
basey += wheight |
|
|
} |
|
|
} |
|
|
} |
|
|
}, 500) |
|
|
}) |
|
|
chainCallback(node, "onMouseLeave", function (e, pos, canvas) { |
|
|
if (timeout) { |
|
|
clearTimeout(timeout) |
|
|
timeout = null |
|
|
} |
|
|
}); |
|
|
} |
|
|
} |
|
|
|
|
|
function fitHeight(node) { |
|
|
node.setSize([node.size[0], node.computeSize([node.size[0], node.size[1]])[1]]) |
|
|
node?.graph?.setDirtyCanvas(true); |
|
|
} |
|
|
function startDraggingItems(node, pointer) { |
|
|
app.canvas.emitBeforeChange() |
|
|
app.canvas.graph?.beforeChange() |
|
|
|
|
|
pointer.finally = () => { |
|
|
app.canvas.isDragging = false |
|
|
app.canvas.graph?.afterChange() |
|
|
app.canvas.emitAfterChange() |
|
|
} |
|
|
app.canvas.processSelect(node, pointer.eDown, true) |
|
|
app.canvas.isDragging = true |
|
|
} |
|
|
function processDraggedItems(e) { |
|
|
if (e.shiftKey || LiteGraph.alwaysSnapToGrid) |
|
|
app.canvas?.graph?.snapToGrid(app.canvas.selectedItems) |
|
|
app.canvas.dirty_canvas = true |
|
|
app.canvas.dirty_bgcanvas = true |
|
|
app.canvas.onNodeMoved?.(findFirstNode(app.canvas.selectedItems)) |
|
|
} |
|
|
function allowDragFromWidget(widget) { |
|
|
widget.onPointerDown = function(pointer, node) { |
|
|
pointer.onDragStart = () => startDraggingItems(node, pointer) |
|
|
pointer.onDragEnd = processDraggedItems |
|
|
app.canvas.dirty_canvas = true |
|
|
return true |
|
|
} |
|
|
} |
|
|
|
|
|
async function uploadFile(file, progressCallback) { |
|
|
try { |
|
|
|
|
|
const body = new FormData(); |
|
|
const i = file.webkitRelativePath.lastIndexOf('/'); |
|
|
const subfolder = file.webkitRelativePath.slice(0,i+1) |
|
|
const new_file = new File([file], file.name, { |
|
|
type: file.type, |
|
|
lastModified: file.lastModified, |
|
|
}); |
|
|
body.append("image", new_file); |
|
|
if (i > 0) { |
|
|
body.append("subfolder", subfolder); |
|
|
} |
|
|
const url = api.apiURL("/upload/image") |
|
|
const resp = await new Promise((resolve) => { |
|
|
let req = new XMLHttpRequest() |
|
|
req.upload.onprogress = (e) => progressCallback?.(e.loaded/e.total) |
|
|
req.onload = () => resolve(req) |
|
|
req.open('post', url, true) |
|
|
req.send(body) |
|
|
}) |
|
|
|
|
|
if (resp.status !== 200) { |
|
|
alert(resp.status + " - " + resp.statusText); |
|
|
} |
|
|
return resp |
|
|
} catch (error) { |
|
|
alert(error); |
|
|
} |
|
|
} |
|
|
|
|
|
function addVAEOutputToggle(nodeType, nodeData) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
this.reject_ue_connection = (input) => input?.name == "vae" |
|
|
}) |
|
|
chainCallback(nodeType.prototype, "onConnectionsChange", function(contype, slot, iscon, linfo) { |
|
|
let slotType = this.inputs[slot]?.type |
|
|
if (contype == LiteGraph.INPUT && slotType == "VAE") { |
|
|
if (iscon && linfo) { |
|
|
if (this.linkTimeout) { |
|
|
clearTimeout(this.linkTimeout) |
|
|
this.linkTimeout = false |
|
|
} else if (this.outputs[0].type == "IMAGE") { |
|
|
this.linkTimeout = setTimeout(() => { |
|
|
if (this.outputs[0].type != "IMAGE") { |
|
|
return |
|
|
} |
|
|
this.linkTimeout = false |
|
|
this.disconnectOutput(0); |
|
|
}, 50) |
|
|
} |
|
|
this.outputs[0].name = 'LATENT'; |
|
|
this.outputs[0].type = 'LATENT'; |
|
|
} else{ |
|
|
if (this.outputs[0].type == "LATENT") { |
|
|
this.linkTimeout = setTimeout(() => { |
|
|
this.linkTimeout = false |
|
|
this.disconnectOutput(0); |
|
|
}, 50) |
|
|
} |
|
|
this.outputs[0].name = "IMAGE"; |
|
|
this.outputs[0].type = "IMAGE"; |
|
|
} |
|
|
} |
|
|
}); |
|
|
} |
|
|
function addVAEInputToggle(nodeType, nodeData) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
this.reject_ue_connection = (input) => input?.name == "vae" |
|
|
}) |
|
|
chainCallback(nodeType.prototype, "onConnectionsChange", function(contype, slot, iscon, linf) { |
|
|
if (contype == LiteGraph.INPUT && slot == 3 && this.inputs[3].type == "VAE") { |
|
|
if (iscon && linf) { |
|
|
if (this.linkTimeout) { |
|
|
clearTimeout(this.linkTimeout) |
|
|
this.linkTimeout = false |
|
|
} else if (this.inputs[0].type == "IMAGE") { |
|
|
this.linkTimeout = setTimeout(() => { |
|
|
|
|
|
if (this.inputs[0].type != "IMAGE") { |
|
|
return |
|
|
} |
|
|
this.linkTimeout = false |
|
|
this.disconnectInput(0); |
|
|
}, 50) |
|
|
} |
|
|
this.inputs[0].type = 'LATENT'; |
|
|
} else { |
|
|
if (this.inputs[0].type == "LATENT") { |
|
|
this.linkTimeout = setTimeout(() => { |
|
|
this.linkTimeout = false |
|
|
this.disconnectInput(0); |
|
|
}, 50) |
|
|
} |
|
|
this.inputs[0].type = "IMAGE"; |
|
|
} |
|
|
} |
|
|
}); |
|
|
} |
|
|
function cloneType(nodeType, nodeData) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
this.changeOutputType = function (new_type) { |
|
|
this.linkTimeout = setTimeout(() => { |
|
|
this.linkTimeout = false |
|
|
if (this.outputs[0].type != new_type) { |
|
|
this.outputs[0].type = new_type |
|
|
|
|
|
if (!this.outputs[0].links) { |
|
|
return |
|
|
} |
|
|
let removed_links = [] |
|
|
for (let link_id of this.outputs[0].links) { |
|
|
let link = app.graph.links[link_id] |
|
|
if (!link) |
|
|
debugger |
|
|
let target_node = app.graph.getNodeById(link.target_id) |
|
|
let target_input = target_node.inputs[link.target_slot] |
|
|
let keep = LiteGraph.isValidConnection(new_type, target_input.type) |
|
|
if (!keep) { |
|
|
link.disconnect(app.graph, 'input') |
|
|
removed_links.push(link_id) |
|
|
} |
|
|
target_node.onConnectionsChange?.(LiteGraph.INPUT, |
|
|
link.target_slot, keep, link, target_input) |
|
|
} |
|
|
this.outputs[0].links = this.outputs[0].links |
|
|
.filter((v) => !removed_links.includes(v)) |
|
|
} |
|
|
}, 50) |
|
|
} |
|
|
this.changeOutputType("VHS_DUMMY_NONE") |
|
|
}); |
|
|
chainCallback(nodeType.prototype, "onConnectionsChange", function(contype, slot, iscon, linf) { |
|
|
if (contype == LiteGraph.INPUT && slot == 0) { |
|
|
let new_type = "VHS_DUMMY_NONE" |
|
|
if (iscon && linf) { |
|
|
new_type = app.graph.getNodeById(linf.origin_id).outputs[linf.origin_slot].type |
|
|
} |
|
|
if (this.linkTimeout) { |
|
|
clearTimeout(this.linkTimeout) |
|
|
} |
|
|
this.changeOutputType(new_type) |
|
|
} |
|
|
}); |
|
|
} |
|
|
|
|
|
function addDateFormatting(nodeType, field, timestamp_widget = false) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const widget = this.widgets.find((w) => w.name === field); |
|
|
widget.serializeValue = () => { |
|
|
return applyTextReplacements(app, widget.value); |
|
|
}; |
|
|
}); |
|
|
} |
|
|
function addTimestampWidget(nodeType, nodeData, targetWidget) { |
|
|
const newWidgets = {}; |
|
|
for (let key in nodeData.input.required) { |
|
|
if (key == targetWidget) { |
|
|
|
|
|
newWidgets["timestamp_directory"] = ["BOOLEAN", {"default": true}] |
|
|
} |
|
|
newWidgets[key] = nodeData.input.required[key]; |
|
|
} |
|
|
nodeDta.input.required = newWidgets; |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function () { |
|
|
const directoryWidget = this.widgets.find((w) => w.name === "directory_name"); |
|
|
const timestampWidget = this.widgets.find((w) => w.name === "timestamp_directory"); |
|
|
directoryWidget.serializeValue = () => { |
|
|
if (timestampWidget.value) { |
|
|
|
|
|
return formatDate("yyyy-MM-ddThh:mm:ss", new Date()); |
|
|
} |
|
|
return directoryWidget.value |
|
|
}; |
|
|
timestampWidget._value = value; |
|
|
Object.definteProperty(timestampWidget, "value", { |
|
|
set : function(value) { |
|
|
this._value = value; |
|
|
directoryWidget.disabled = value; |
|
|
}, |
|
|
get : function() { |
|
|
return this._value; |
|
|
} |
|
|
}); |
|
|
}); |
|
|
} |
|
|
function initializeLoadFormat(nodeType, nodeData) { |
|
|
if (!nodeData?.input?.optional?.format) { |
|
|
return |
|
|
} |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
let node = this |
|
|
let formatWidget = this.widgets.find((w) => w.name === "format") |
|
|
formatWidget.options.formats = nodeData.input.optional.format[1].formats |
|
|
let base = {} |
|
|
for (let widget of this.widgets) { |
|
|
if (['force_rate', 'custom_width', 'custom_height', |
|
|
'frame_load_cap'].includes(widget.name)) { |
|
|
|
|
|
base[widget.name] = widget.options |
|
|
} |
|
|
} |
|
|
chainCallback(formatWidget, "callback", function(value) { |
|
|
let format = this.options.formats[value] |
|
|
if (!format) { |
|
|
return |
|
|
} |
|
|
if ('target_rate' in format) { |
|
|
format.force_rate = {'reset': format.target_rate} |
|
|
} |
|
|
if ('dim' in format) { |
|
|
format.custom_width = {'step': format.dim[0], 'mod': format.dim[1]} |
|
|
format.custom_height = {'step': format.dim[0], 'mod': format.dim[1]} |
|
|
if (format.dim[2]) { |
|
|
format.custom_width.reset = format.dim[2] |
|
|
} |
|
|
if (format.dim[3]) { |
|
|
format.custom_height.reset = format.dim[3] |
|
|
} |
|
|
} |
|
|
if ('frames' in format) { |
|
|
format.frame_load_cap = {'step': format.frames[0], 'mod': format.frames[1]} |
|
|
} |
|
|
for (let widget of node.widgets) { |
|
|
if (widget.name in base) { |
|
|
let wasDefault = widget.options?.reset == widget.value |
|
|
widget.options = Object.assign({}, base[widget.name], format[widget.name]) |
|
|
if (wasDefault && widget.options.reset != undefined) { |
|
|
widget.value = widget.options.reset |
|
|
} |
|
|
widget.callback(widget.value) |
|
|
} |
|
|
} |
|
|
|
|
|
}); |
|
|
let capWidget = this.widgets.find((w) => w.name === "frame_load_cap") |
|
|
capWidget.annotation = (value, width) => { |
|
|
let max_frames = this.video_query?.loaded?.frames |
|
|
if (!max_frames || value && value < max_frames) { |
|
|
return |
|
|
} |
|
|
let format = formatWidget.options.formats[formatWidget.value] |
|
|
const div = format?.frames?.[0] ?? 1 |
|
|
const mod = format?.frames?.[1] ?? 0 |
|
|
let loadable_frames = max_frames |
|
|
if ((max_frames % div) != mod) { |
|
|
loadable_frames = ((max_frames - mod)/div|0) * div + mod |
|
|
} |
|
|
return loadable_frames + "\u21FD" |
|
|
} |
|
|
let rateWidget = this.widgets.find((w) => w.name === "force_rate") |
|
|
rateWidget.annotation = (value, width) => { |
|
|
if (value == 0 && this.video_query?.source?.fps != undefined) { |
|
|
return roundToPrecision(this.video_query.source.fps, 2) + "\u21FD" |
|
|
} |
|
|
} |
|
|
}); |
|
|
} |
|
|
|
|
|
function addUploadWidget(nodeType, nodeData, widgetName, type="video") { |
|
|
let accept = {'video': ["video/webm","video/mp4","video/x-matroska","image/gif"], |
|
|
'audio': ["audio/mpeg","audio/wav","audio/x-wav","audio/ogg"]} |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const node = this |
|
|
const pathWidget = this.widgets.find((w) => w.name === widgetName); |
|
|
const fileInput = document.createElement("input"); |
|
|
chainCallback(this, "onRemoved", () => { |
|
|
fileInput?.remove(); |
|
|
}); |
|
|
if (type == "folder") { |
|
|
Object.assign(fileInput, { |
|
|
type: "file", |
|
|
style: "display: none", |
|
|
webkitdirectory: true, |
|
|
onchange: async () => { |
|
|
const directory = fileInput.files[0].webkitRelativePath; |
|
|
const i = directory.lastIndexOf('/'); |
|
|
if (i <= 0) { |
|
|
throw "No directory found"; |
|
|
} |
|
|
const path = directory.slice(0,directory.lastIndexOf('/')) |
|
|
if (pathWidget.options.values.includes(path)) { |
|
|
alert("A folder of the same name already exists"); |
|
|
return; |
|
|
} |
|
|
let successes = 0; |
|
|
const onProg = (p) => this.progress = (successes + p) / fileInput.files.length |
|
|
for(const file of fileInput.files) { |
|
|
if ((await uploadFile(file, onProg)).status == 200) { |
|
|
successes++; |
|
|
} else { |
|
|
this.progress = undefined |
|
|
|
|
|
|
|
|
|
|
|
if (successes > 0) { |
|
|
break |
|
|
} else { |
|
|
return; |
|
|
} |
|
|
} |
|
|
} |
|
|
this.progress = undefined |
|
|
pathWidget.options.values.push(path); |
|
|
pathWidget.value = path; |
|
|
if (pathWidget.callback) { |
|
|
pathWidget.callback(path) |
|
|
} |
|
|
}, |
|
|
}); |
|
|
} else { |
|
|
let accept = {'video': ["video/webm","video/mp4","video/x-matroska","image/gif"], |
|
|
'audio': ["audio/mpeg","audio/wav","audio/x-wav","audio/ogg"]}[type] |
|
|
async function doUpload(file) { |
|
|
let resp = await uploadFile(file, (p) => node.progress = p) |
|
|
node.progress = undefined |
|
|
if (resp.status != 200) { |
|
|
return false |
|
|
} |
|
|
const filename = JSON.parse(resp.responseText).name; |
|
|
pathWidget.options.values.push(filename); |
|
|
pathWidget.value = filename; |
|
|
if (pathWidget.callback) { |
|
|
pathWidget.callback(filename) |
|
|
} |
|
|
return true |
|
|
} |
|
|
Object.assign(fileInput, { |
|
|
type: "file", |
|
|
accept: accept.join(','), |
|
|
style: "display: none", |
|
|
onchange: async () => { |
|
|
if (fileInput.files.length) { |
|
|
return await doUpload(fileInput.files[0]) |
|
|
} |
|
|
}, |
|
|
}); |
|
|
this.onDragOver = (e) => !!e?.dataTransfer?.types?.includes?.('Files') |
|
|
this.onDragDrop = async function(e) { |
|
|
if (!e?.dataTransfer?.types?.includes?.('Files')) { |
|
|
return false |
|
|
} |
|
|
|
|
|
const item = e.dataTransfer?.files?.[0] |
|
|
if (accept.includes(item?.type)) { |
|
|
return await doUpload(item) |
|
|
} |
|
|
return false |
|
|
} |
|
|
} |
|
|
document.body.append(fileInput); |
|
|
let uploadWidget = this.addWidget("button", "choose " + type + " to upload", "image", () => { |
|
|
|
|
|
app.canvas.node_widget = null |
|
|
|
|
|
fileInput.click(); |
|
|
}); |
|
|
uploadWidget.options.serialize = false; |
|
|
|
|
|
|
|
|
}); |
|
|
} |
|
|
function addAudioPreview(nodeType, isInput=true) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
var element = document.createElement("audio"); |
|
|
element.controls = true |
|
|
const previewNode = this; |
|
|
var previewWidget = this.addDOMWidget("audiopreview", "preview", element, { |
|
|
serialize: false, |
|
|
hideOnZoom: true, |
|
|
getValue() { |
|
|
return element.value; |
|
|
}, |
|
|
setValue(v) { |
|
|
element.value = v; |
|
|
}, |
|
|
}); |
|
|
previewWidget.computeSize = function(width) { |
|
|
return [width, 50]; |
|
|
} |
|
|
var timeout = null; |
|
|
this.updateParameters = (params, force_update) => { |
|
|
if (!previewWidget.value.params) { |
|
|
if(typeof(previewWidget.value) != 'object') { |
|
|
previewWidget.value = {} |
|
|
} |
|
|
previewWidget.value.params = {} |
|
|
} |
|
|
Object.assign(previewWidget.value.params, params) |
|
|
if (!force_update && |
|
|
app.ui.settings.getSettingValue("VHS.AdvancedPreviews") == 'Never') { |
|
|
return; |
|
|
} |
|
|
if (timeout) { |
|
|
clearTimeout(timeout); |
|
|
} |
|
|
if (force_update) { |
|
|
previewWidget.updateSource(); |
|
|
} else { |
|
|
timeout = setTimeout(() => previewWidget.updateSource(),100); |
|
|
} |
|
|
}; |
|
|
previewWidget.updateSource = function () { |
|
|
if (this.value.params == undefined) { |
|
|
return; |
|
|
} |
|
|
let params = {} |
|
|
let advp = app.ui.settings.getSettingValue("VHS.AdvancedPreviews") |
|
|
if (advp == 'Never') { |
|
|
advp = false |
|
|
} else if (advp == 'Input Only') { |
|
|
advp = isInput |
|
|
} else { |
|
|
advp = true |
|
|
} |
|
|
Object.assign(params, this.value.params); |
|
|
params.timestamp = Date.now() |
|
|
if (!advp) { |
|
|
element.src = api.apiURL('/view?' + new URLSearchParams(params)); |
|
|
} else { |
|
|
params.deadline = app.ui.settings.getSettingValue("VHS.AdvancedPreviewsDeadline") |
|
|
element.src = api.apiURL('/vhs/viewaudio?' + new URLSearchParams(params)); |
|
|
} |
|
|
} |
|
|
previewWidget.callback = previewWidget.updateSource |
|
|
|
|
|
|
|
|
|
|
|
function update(key) { |
|
|
return function(value) { |
|
|
let params = {} |
|
|
params[key] = this.value |
|
|
previewNode?.updateParameters(params) |
|
|
} |
|
|
} |
|
|
let widgetMap = { 'seek_seconds': 'start_time', 'duration': 'duration', |
|
|
'start_time': 'start_time' } |
|
|
for (let widget of this.widgets) { |
|
|
if (widget.name in widgetMap) { |
|
|
if (typeof(widgetMap[widget.name]) == 'function') { |
|
|
chainCallback(widget, "callback", widgetMap[widget.name]); |
|
|
} else { |
|
|
chainCallback(widget, "callback", update(widgetMap[widget.name])) |
|
|
} |
|
|
} |
|
|
if (widget.type != "button") { |
|
|
widget.callback?.(widget.value) |
|
|
} |
|
|
} |
|
|
}); |
|
|
} |
|
|
|
|
|
function addVideoPreview(nodeType, isInput=true) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
var element = document.createElement("div"); |
|
|
const previewNode = this; |
|
|
var previewWidget = this.addDOMWidget("videopreview", "preview", element, { |
|
|
serialize: false, |
|
|
hideOnZoom: false, |
|
|
getValue() { |
|
|
return element.value; |
|
|
}, |
|
|
setValue(v) { |
|
|
element.value = v; |
|
|
}, |
|
|
}); |
|
|
allowDragFromWidget(previewWidget) |
|
|
previewWidget.computeSize = function(width) { |
|
|
if (this.aspectRatio && !this.parentEl.hidden) { |
|
|
let height = (previewNode.size[0]-20)/ this.aspectRatio + 10; |
|
|
if (!(height > 0)) { |
|
|
height = 0; |
|
|
} |
|
|
this.computedHeight = height + 10; |
|
|
return [width, height]; |
|
|
} |
|
|
return [width, -4]; |
|
|
} |
|
|
element.addEventListener('contextmenu', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousedown_callback(e) |
|
|
}, true); |
|
|
element.addEventListener('pointerdown', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousedown_callback(e) |
|
|
}, true); |
|
|
element.addEventListener('mousewheel', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousewheel_callback(e) |
|
|
}, true); |
|
|
element.addEventListener('pointermove', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousemove_callback(e) |
|
|
}, true); |
|
|
element.addEventListener('pointerup', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mouseup_callback(e) |
|
|
}, true); |
|
|
element.addEventListener('dragover', (e) => { |
|
|
|
|
|
e.preventDefault(); |
|
|
e.dataTransfer.dropEffect = "copy"; |
|
|
app.dragOverNode = this |
|
|
}) |
|
|
previewWidget.value = {hidden: false, paused: false, params: {}, |
|
|
muted: app.ui.settings.getSettingValue("VHS.DefaultMute")} |
|
|
previewWidget.parentEl = document.createElement("div"); |
|
|
previewWidget.parentEl.className = "vhs_preview"; |
|
|
previewWidget.parentEl.style['width'] = "100%" |
|
|
element.appendChild(previewWidget.parentEl); |
|
|
previewWidget.videoEl = document.createElement("video"); |
|
|
previewWidget.videoEl.controls = false; |
|
|
previewWidget.videoEl.loop = true; |
|
|
previewWidget.videoEl.muted = true; |
|
|
previewWidget.videoEl.style['width'] = "100%" |
|
|
previewWidget.videoEl.addEventListener("loadedmetadata", () => { |
|
|
|
|
|
previewWidget.aspectRatio = previewWidget.videoEl.videoWidth / previewWidget.videoEl.videoHeight; |
|
|
fitHeight(this); |
|
|
}); |
|
|
previewWidget.videoEl.addEventListener("error", () => { |
|
|
|
|
|
previewWidget.parentEl.hidden = true; |
|
|
fitHeight(this); |
|
|
}); |
|
|
previewWidget.videoEl.onmouseenter = () => { |
|
|
previewWidget.videoEl.muted = previewWidget.value.muted |
|
|
}; |
|
|
previewWidget.videoEl.onmouseleave = () => { |
|
|
previewWidget.videoEl.muted = true; |
|
|
}; |
|
|
|
|
|
previewWidget.imgEl = document.createElement("img"); |
|
|
previewWidget.imgEl.style['width'] = "100%" |
|
|
previewWidget.imgEl.hidden = true; |
|
|
previewWidget.imgEl.onload = () => { |
|
|
previewWidget.aspectRatio = previewWidget.imgEl.naturalWidth / previewWidget.imgEl.naturalHeight; |
|
|
fitHeight(this); |
|
|
}; |
|
|
previewWidget.parentEl.appendChild(previewWidget.videoEl) |
|
|
previewWidget.parentEl.appendChild(previewWidget.imgEl) |
|
|
var timeout = null; |
|
|
this.updateParameters = (params, force_update) => { |
|
|
if (!previewWidget.value.params) { |
|
|
if(typeof(previewWidget.value) != 'object') { |
|
|
previewWidget.value = {hidden: false, paused: false} |
|
|
} |
|
|
previewWidget.value.params = {} |
|
|
} |
|
|
Object.assign(previewWidget.value.params, params) |
|
|
if (!force_update && |
|
|
app.ui.settings.getSettingValue("VHS.AdvancedPreviews") == 'Never') { |
|
|
return; |
|
|
} |
|
|
if (timeout) { |
|
|
clearTimeout(timeout); |
|
|
} |
|
|
if (force_update) { |
|
|
previewWidget.updateSource(); |
|
|
} else { |
|
|
timeout = setTimeout(() => previewWidget.updateSource(),100); |
|
|
} |
|
|
}; |
|
|
previewWidget.updateSource = function () { |
|
|
if (this.value.params == undefined) { |
|
|
return; |
|
|
} |
|
|
let params = {} |
|
|
let advp = app.ui.settings.getSettingValue("VHS.AdvancedPreviews") |
|
|
if (advp == 'Never') { |
|
|
advp = false |
|
|
} else if (advp == 'Input Only') { |
|
|
advp = isInput |
|
|
} else { |
|
|
advp = true |
|
|
} |
|
|
Object.assign(params, this.value.params); |
|
|
params.timestamp = Date.now() |
|
|
this.parentEl.hidden = this.value.hidden; |
|
|
if (params.format?.split('/')[0] == 'video' |
|
|
|| advp && (params.format?.split('/')[1] == 'gif') |
|
|
|| params.format == 'folder') { |
|
|
|
|
|
this.videoEl.autoplay = !this.value.paused && !this.value.hidden; |
|
|
if (!advp) { |
|
|
this.videoEl.src = api.apiURL('/view?' + new URLSearchParams(params)); |
|
|
} else { |
|
|
let target_width = (previewNode.size[0]-20)*2 || 256; |
|
|
let minWidth = app.ui.settings.getSettingValue("VHS.AdvancedPreviewsMinWidth") |
|
|
if (target_width < minWidth) { |
|
|
target_width = minWidth |
|
|
} |
|
|
if (!params.custom_width || !params.custom_height) { |
|
|
params.force_size = target_width+"x?" |
|
|
} else { |
|
|
let ar = params.custom_width/params.custom_height |
|
|
params.force_size = target_width+"x"+(target_width/ar) |
|
|
} |
|
|
params.deadline = app.ui.settings.getSettingValue("VHS.AdvancedPreviewsDeadline") |
|
|
this.videoEl.src = api.apiURL('/vhs/viewvideo?' + new URLSearchParams(params)); |
|
|
} |
|
|
this.videoEl.hidden = false; |
|
|
this.imgEl.hidden = true; |
|
|
} else if (params.format?.split('/')[0] == 'image'){ |
|
|
|
|
|
this.imgEl.src = api.apiURL('/view?' + new URLSearchParams(params)); |
|
|
this.videoEl.hidden = true; |
|
|
this.imgEl.hidden = false; |
|
|
} |
|
|
delete previewNode.video_query |
|
|
const doQuery = async () => { |
|
|
if (!previewWidget?.value?.params?.filename) { |
|
|
return |
|
|
} |
|
|
let qurl = api.apiURL('/vhs/queryvideo?' + new URLSearchParams(previewWidget.value.params)) |
|
|
let query = undefined |
|
|
try { |
|
|
let query_res = await fetch(qurl) |
|
|
query = await query_res.json() |
|
|
} catch(e) { |
|
|
return |
|
|
} |
|
|
previewNode.video_query = query |
|
|
} |
|
|
doQuery() |
|
|
} |
|
|
previewWidget.callback = previewWidget.updateSource |
|
|
previewWidget.parentEl.appendChild(previewWidget.videoEl) |
|
|
previewWidget.parentEl.appendChild(previewWidget.imgEl) |
|
|
}); |
|
|
} |
|
|
let copiedPath = undefined |
|
|
function addPreviewOptions(nodeType) { |
|
|
chainCallback(nodeType.prototype, "getExtraMenuOptions", function(_, options) { |
|
|
|
|
|
|
|
|
|
|
|
let optNew = [] |
|
|
const previewWidget = this.widgets.find((w) => w.name === "videopreview"); |
|
|
|
|
|
let url = null |
|
|
if (previewWidget.videoEl?.hidden == false && previewWidget.videoEl.src) { |
|
|
if (['input', 'output', 'temp'].includes(previewWidget.value.params.type)) { |
|
|
|
|
|
url = api.apiURL('/view?' + new URLSearchParams(previewWidget.value.params)); |
|
|
|
|
|
url = url.replace('%2503d', '001') |
|
|
} |
|
|
} else if (previewWidget.imgEl?.hidden == false && previewWidget.imgEl.src) { |
|
|
url = previewWidget.imgEl.src; |
|
|
url = new URL(url); |
|
|
} |
|
|
if (this.video_query?.source) { |
|
|
let info_string = this.video_query.source.size.join('x') + |
|
|
'@' + this.video_query.source.fps + 'fps ' + |
|
|
this.video_query.source.frames + 'frames' |
|
|
optNew.push({content: info_string, disabled: true}) |
|
|
} |
|
|
if (url) { |
|
|
optNew.push( |
|
|
{ |
|
|
content: "Open preview", |
|
|
callback: () => { |
|
|
window.open(url, "_blank") |
|
|
}, |
|
|
}, |
|
|
{ |
|
|
content: "Save preview", |
|
|
callback: () => { |
|
|
const a = document.createElement("a"); |
|
|
a.href = url; |
|
|
a.setAttribute("download", previewWidget.value.params.filename); |
|
|
document.body.append(a); |
|
|
a.click(); |
|
|
requestAnimationFrame(() => a.remove()); |
|
|
}, |
|
|
} |
|
|
); |
|
|
if (previewWidget.value.params.fullpath) { |
|
|
copiedPath = previewWidget.value.params.fullpath |
|
|
const blob = new Blob([previewWidget.value.params.fullpath], |
|
|
{ type: 'text/plain'}) |
|
|
optNew.push({ |
|
|
content: "Copy output filepath", |
|
|
callback: async () => { |
|
|
await navigator.clipboard.write([ |
|
|
new ClipboardItem({ |
|
|
'text/plain': blob |
|
|
})])} |
|
|
}); |
|
|
} |
|
|
if (previewWidget.value.params.workflow) { |
|
|
let wParams = {...previewWidget.value.params, |
|
|
filename: previewWidget.value.params.workflow} |
|
|
let wUrl = api.apiURL('/view?' + new URLSearchParams(wParams)); |
|
|
optNew.push({ |
|
|
content: "Save workflow image", |
|
|
callback: () => { |
|
|
const a = document.createElement("a"); |
|
|
a.href = wUrl; |
|
|
a.setAttribute("download", previewWidget.value.params.workflow); |
|
|
document.body.append(a); |
|
|
a.click(); |
|
|
requestAnimationFrame(() => a.remove()); |
|
|
} |
|
|
}); |
|
|
} |
|
|
} |
|
|
const PauseDesc = (previewWidget.value.paused ? "Resume" : "Pause") + " preview"; |
|
|
if(previewWidget.videoEl.hidden == false) { |
|
|
optNew.push({content: PauseDesc, callback: () => { |
|
|
|
|
|
|
|
|
|
|
|
if(previewWidget.value.paused) { |
|
|
previewWidget.videoEl?.play(); |
|
|
} else { |
|
|
previewWidget.videoEl?.pause(); |
|
|
} |
|
|
previewWidget.value.paused = !previewWidget.value.paused; |
|
|
}}); |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const visDesc = (previewWidget.value.hidden ? "Show" : "Hide") + " preview"; |
|
|
optNew.push({content: visDesc, callback: () => { |
|
|
if (!previewWidget.videoEl.hidden && !previewWidget.value.hidden) { |
|
|
previewWidget.videoEl.pause(); |
|
|
} else if (previewWidget.value.hidden && !previewWidget.videoEl.hidden && !previewWidget.value.paused) { |
|
|
previewWidget.videoEl.play(); |
|
|
} |
|
|
previewWidget.value.hidden = !previewWidget.value.hidden; |
|
|
previewWidget.parentEl.hidden = previewWidget.value.hidden; |
|
|
fitHeight(this); |
|
|
|
|
|
}}); |
|
|
optNew.push({content: "Sync preview", callback: () => { |
|
|
|
|
|
|
|
|
for (let p of document.getElementsByClassName("vhs_preview")) { |
|
|
for (let child of p.children) { |
|
|
if (child.tagName == "VIDEO") { |
|
|
child.currentTime=0; |
|
|
} else if (child.tagName == "IMG") { |
|
|
child.src = child.src; |
|
|
} |
|
|
} |
|
|
} |
|
|
}}); |
|
|
const muteDesc = (previewWidget.value.muted ? "Unmute" : "Mute") + " Preview" |
|
|
optNew.push({content: muteDesc, callback: () => { |
|
|
previewWidget.value.muted = !previewWidget.value.muted |
|
|
}}) |
|
|
if(options.length > 0 && options[0] != null && optNew.length > 0) { |
|
|
optNew.push(null); |
|
|
} |
|
|
options.unshift(...optNew); |
|
|
}); |
|
|
} |
|
|
function addFormatWidgets(nodeType, nodeData) { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
var formatWidget = null; |
|
|
var formatWidgetIndex = -1; |
|
|
for(let i = 0; i < this.widgets.length; i++) { |
|
|
if (this.widgets[i].name === "format"){ |
|
|
formatWidget = this.widgets[i]; |
|
|
formatWidgetIndex = i+1; |
|
|
break |
|
|
} |
|
|
} |
|
|
let formatWidgetsCount = 0; |
|
|
chainCallback(formatWidget, "callback", (value) => { |
|
|
const formats = (LiteGraph.registered_node_types[this.type] |
|
|
?.nodeData?.input?.required?.format?.[1]?.formats) |
|
|
let newWidgets = []; |
|
|
if (formats?.[value]) { |
|
|
let formatWidgets = formats[value] |
|
|
for (let wDef of formatWidgets) { |
|
|
let type = wDef[2]?.widgetType ?? wDef[1] |
|
|
if (Array.isArray(type)) { |
|
|
type = "COMBO" |
|
|
} |
|
|
app.widgets[type](this, wDef[0], wDef.slice(1), app) |
|
|
let w = this.widgets.pop() |
|
|
w.config = wDef.slice(1) |
|
|
newWidgets.push(w) |
|
|
} |
|
|
} |
|
|
let removed = this.widgets.splice(formatWidgetIndex, |
|
|
formatWidgetsCount, ...newWidgets); |
|
|
let newNames = new Set(newWidgets.map((w) => w.name)) |
|
|
for (let w of removed) { |
|
|
w?.onRemove?.() |
|
|
if (w.name in newNames) { |
|
|
continue |
|
|
} |
|
|
|
|
|
let slot = this.inputs.findIndex((i) => i.name == w.name) |
|
|
if (slot >= 0) { |
|
|
this.removeInput(slot) |
|
|
} |
|
|
} |
|
|
for (let w of newWidgets) { |
|
|
let existingInput = this.inputs.find((i) => i.name == w.name) |
|
|
if (existingInput) { |
|
|
setWidgetConfig(existingInput, w.config) |
|
|
|
|
|
} else { |
|
|
|
|
|
this.addInput(w.name, w.config[0], {widget: {name: w.name}}) |
|
|
} |
|
|
} |
|
|
fitHeight(this); |
|
|
formatWidgetsCount = newWidgets.length; |
|
|
}); |
|
|
}); |
|
|
} |
|
|
function addLoadCommon(nodeType, nodeData) { |
|
|
addVideoPreview(nodeType); |
|
|
initializeLoadFormat(nodeType, nodeData) |
|
|
addPreviewOptions(nodeType); |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
|
|
|
const node = this |
|
|
function update(key) { |
|
|
return function(value) { |
|
|
let params = {} |
|
|
params[key] = this.value |
|
|
node?.updateParameters(params) |
|
|
} |
|
|
} |
|
|
let prior_ar = -2 |
|
|
const widthWidget = this.widgets.find((w) => w.name === "custom_width"); |
|
|
const heightWidget = this.widgets.find((w) => w.name === "custom_height"); |
|
|
function updateAR(value) { |
|
|
let new_ar = -1 |
|
|
if (widthWidget.value & heightWidget.value) { |
|
|
new_ar = widthWidget.value / heightWidget.value |
|
|
} |
|
|
if (new_ar != prior_ar) { |
|
|
node?.updateParameters({'custom_width': widthWidget.value, |
|
|
'custom_height': heightWidget.value}) |
|
|
prior_ar = new_ar |
|
|
} |
|
|
} |
|
|
const offsetWidget = this.widgets.find((w) => w.name === "start_time"); |
|
|
if (offsetWidget) { |
|
|
Object.defineProperty(offsetWidget.options, "step", { |
|
|
set : (value) => {}, |
|
|
get : () => { |
|
|
return 1 / (this.video_query?.loaded?.fps ?? 1) |
|
|
} |
|
|
}) |
|
|
} |
|
|
let widgetMap = {'frame_load_cap': 'frame_load_cap', |
|
|
'skip_first_frames': 'skip_first_frames', 'select_every_nth': 'select_every_nth', |
|
|
'start_time': 'start_time', 'force_rate': 'force_rate', |
|
|
'custom_width': updateAR, 'custom_height': updateAR, |
|
|
'image_load_cap': 'image_load_cap', 'skip_first_images': 'skip_first_images' |
|
|
} |
|
|
for (let widget of this.widgets) { |
|
|
if (widget.name in widgetMap) { |
|
|
if (typeof(widgetMap[widget.name]) == 'function') { |
|
|
chainCallback(widget, "callback", widgetMap[widget.name]); |
|
|
} else { |
|
|
chainCallback(widget, "callback", update(widgetMap[widget.name])) |
|
|
} |
|
|
} |
|
|
if (widget.type != "button") { |
|
|
widget.callback?.(widget.value) |
|
|
} |
|
|
} |
|
|
}); |
|
|
} |
|
|
|
|
|
function path_stem(path) { |
|
|
let i = path.lastIndexOf("/"); |
|
|
if (i >= 0) { |
|
|
return [path.slice(0,i+1),path.slice(i+1)]; |
|
|
} |
|
|
return ["",path]; |
|
|
} |
|
|
function searchBox(event, [x,y], node) { |
|
|
|
|
|
if (this.prompt) |
|
|
return; |
|
|
this.prompt = true; |
|
|
|
|
|
let pathWidget = this; |
|
|
let dialog = document.createElement("div"); |
|
|
dialog.className = "litegraph litesearchbox graphdialog rounded" |
|
|
dialog.innerHTML = '<span class="name">Path</span> <input autofocus="" type="text" class="value"><button class="rounded">OK</button><div class="helper"></div>' |
|
|
dialog.close = () => { |
|
|
dialog.remove(); |
|
|
} |
|
|
document.body.append(dialog); |
|
|
if (app.canvas.ds.scale > 1) { |
|
|
dialog.style.transform = "scale(" + app.canvas.ds.scale + ")"; |
|
|
} |
|
|
var name_element = dialog.querySelector(".name"); |
|
|
var input = dialog.querySelector(".value"); |
|
|
var options_element = dialog.querySelector(".helper"); |
|
|
input.value = pathWidget.value; |
|
|
|
|
|
var timeout = null; |
|
|
let last_path = null; |
|
|
let extensions = pathWidget.options.vhs_path_extensions |
|
|
|
|
|
input.addEventListener("keydown", (e) => { |
|
|
dialog.is_modified = true; |
|
|
if (e.keyCode == 27) { |
|
|
|
|
|
dialog.close(); |
|
|
} else if (e.keyCode == 13 && e.target.localName != "textarea") { |
|
|
pathWidget.value = input.value; |
|
|
if (pathWidget.callback) { |
|
|
pathWidget.callback(pathWidget.value); |
|
|
} |
|
|
dialog.close(); |
|
|
} else { |
|
|
if (e.keyCode == 9) { |
|
|
|
|
|
input.value = last_path + options_element.firstChild.innerText; |
|
|
e.preventDefault(); |
|
|
e.stopPropagation(); |
|
|
} else if (e.ctrlKey && (e.keyCode == 87 || e.keyCode == 66)) { |
|
|
|
|
|
|
|
|
input.value = path_stem(input.value.slice(0,-1))[0] |
|
|
e.preventDefault(); |
|
|
e.stopPropagation(); |
|
|
} else if (e.ctrlKey && e.keyCode == 71) { |
|
|
|
|
|
|
|
|
e.preventDefault(); |
|
|
e.stopPropagation(); |
|
|
extensions = undefined |
|
|
last_path = null; |
|
|
} |
|
|
if (timeout) { |
|
|
clearTimeout(timeout); |
|
|
} |
|
|
timeout = setTimeout(updateOptions, 10); |
|
|
return; |
|
|
} |
|
|
this.prompt=false; |
|
|
e.preventDefault(); |
|
|
e.stopPropagation(); |
|
|
}); |
|
|
|
|
|
var button = dialog.querySelector("button"); |
|
|
button.addEventListener("click", (e) => { |
|
|
pathWidget.value = input.value; |
|
|
if (pathWidget.callback) { |
|
|
pathWidget.callback(pathWidget.value); |
|
|
} |
|
|
|
|
|
node.graph.setDirtyCanvas(true); |
|
|
dialog.close(); |
|
|
this.prompt = false; |
|
|
}); |
|
|
var rect = app.canvas.canvas.getBoundingClientRect(); |
|
|
var offsetx = -20; |
|
|
var offsety = -20; |
|
|
if (rect) { |
|
|
offsetx -= rect.left; |
|
|
offsety -= rect.top; |
|
|
} |
|
|
|
|
|
if (event) { |
|
|
dialog.style.left = event.clientX + offsetx + "px"; |
|
|
dialog.style.top = event.clientY + offsety + "px"; |
|
|
} else { |
|
|
dialog.style.left = canvas.width * 0.5 + offsetx + "px"; |
|
|
dialog.style.top = canvas.height * 0.5 + offsety + "px"; |
|
|
} |
|
|
|
|
|
let options = [] |
|
|
function addResult(name, isDir) { |
|
|
let el = document.createElement("div"); |
|
|
el.innerText = name; |
|
|
el.className = "litegraph lite-search-item"; |
|
|
if (isDir) { |
|
|
el.className += " is-dir"; |
|
|
el.addEventListener("click", (e) => { |
|
|
input.value = last_path+name |
|
|
if (timeout) { |
|
|
clearTimeout(timeout); |
|
|
} |
|
|
timeout = setTimeout(updateOptions, 10); |
|
|
}); |
|
|
} else { |
|
|
el.addEventListener("click", (e) => { |
|
|
pathWidget.value = last_path+name; |
|
|
if (pathWidget.callback) { |
|
|
pathWidget.callback(pathWidget.value); |
|
|
} |
|
|
dialog.close(); |
|
|
pathWidget.prompt = false; |
|
|
}); |
|
|
} |
|
|
options_element.appendChild(el); |
|
|
} |
|
|
async function updateOptions() { |
|
|
timeout = null; |
|
|
let [path, remainder] = path_stem(input.value); |
|
|
if (last_path != path) { |
|
|
|
|
|
let params = {path : path} |
|
|
if (extensions) { |
|
|
params.extensions = extensions |
|
|
} |
|
|
let optionsURL = api.apiURL('/vhs/getpath?' + new URLSearchParams(params)); |
|
|
try { |
|
|
let resp = await fetch(optionsURL); |
|
|
options = await resp.json(); |
|
|
options = options.map((o) => o.replace('.','\0')) |
|
|
options = options.sort() |
|
|
options = options.map((o) => o.replace('\0','.')) |
|
|
} catch(e) { |
|
|
options = [] |
|
|
} |
|
|
last_path = path; |
|
|
} |
|
|
options_element.innerHTML = ''; |
|
|
|
|
|
for (let option of options) { |
|
|
if (option.startsWith(remainder)) { |
|
|
let isDir = option.endsWith('/') |
|
|
addResult(option, isDir); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
setTimeout(async function() { |
|
|
input.focus(); |
|
|
await updateOptions(); |
|
|
}, 10); |
|
|
|
|
|
return dialog; |
|
|
} |
|
|
function button_action(widget) { |
|
|
if ( |
|
|
widget.options?.reset == undefined && |
|
|
widget.options?.disable == undefined |
|
|
) { |
|
|
return 'None' |
|
|
} |
|
|
if ( |
|
|
widget.options.reset != undefined && |
|
|
widget.value != widget.options.reset |
|
|
) { |
|
|
return 'Reset' |
|
|
} |
|
|
if ( |
|
|
widget.options.disable != undefined && |
|
|
widget.value != widget.options.disable |
|
|
) { |
|
|
return 'Disable' |
|
|
} |
|
|
if (widget.options.reset != undefined) { |
|
|
return 'No Reset' |
|
|
} |
|
|
return 'No Disable' |
|
|
} |
|
|
function fitText(ctx, text, maxLength) { |
|
|
if (maxLength <= 0) { |
|
|
return ['', 0] |
|
|
} |
|
|
let fullLength = ctx.measureText(text).width |
|
|
if (fullLength < maxLength) { |
|
|
return [text, fullLength] |
|
|
} |
|
|
|
|
|
let cutoff = maxLength / fullLength * text.length | 0 |
|
|
let shortened = text.slice(0, Math.max(0, cutoff - 2)) + '…' |
|
|
return [shortened, ctx.measureText(shortened).width] |
|
|
} |
|
|
function fitPath(ctx, path, maxLength) { |
|
|
let fullLength = ctx.measureText(path).width |
|
|
if (fullLength < maxLength) { |
|
|
return [path, fullLength] |
|
|
} |
|
|
|
|
|
let len = (maxLength / fullLength * path.length | 0) - 1 |
|
|
|
|
|
let displayPath = '' |
|
|
let filename = path_stem(path)[1] |
|
|
if (filename.length > len-2) { |
|
|
|
|
|
displayPath = filename.substr(0,len); |
|
|
} else { |
|
|
|
|
|
let isAbs = path[0] == '/'; |
|
|
let partial = path.substr(path.length - (isAbs ? len-2:len-1)) |
|
|
let cutoff = partial.indexOf('/'); |
|
|
if (cutoff < 0) { |
|
|
|
|
|
displayPath = path.substr(path.length-len); |
|
|
} else { |
|
|
displayPath = (isAbs ? '/…':'…') + partial.substr(cutoff); |
|
|
} |
|
|
} |
|
|
return [displayPath, ctx.measureText(displayPath).width] |
|
|
} |
|
|
function roundToPrecision(num, precision) { |
|
|
let strnum = Number(num).toFixed(precision) |
|
|
let deci = strnum.indexOf('.') |
|
|
if (deci > 0) { |
|
|
let i = strnum.length - 1 |
|
|
while (i > deci && strnum[i] == '0') { |
|
|
i-- |
|
|
} |
|
|
if (i == deci) { |
|
|
i-- |
|
|
} |
|
|
return strnum.slice(0, i+1) |
|
|
} |
|
|
return strnum |
|
|
} |
|
|
function inner_value_change(widget, value, node, pos) { |
|
|
widget.value = value |
|
|
if (widget.options?.property && widget.options.property in node.properties) { |
|
|
node.setProperty(widget.options.property, value) |
|
|
} |
|
|
if (widget.callback) { |
|
|
widget.callback(widget.value, app.canvas, node, event) |
|
|
} |
|
|
} |
|
|
function drawAnnotated(ctx, node, widget_width, y, H) { |
|
|
const litegraph_base = LiteGraph |
|
|
const show_text = app.canvas.ds.scale >= (app.canvas.low_quality_zoom_threshold ?? 0.5) |
|
|
const margin = 15 |
|
|
ctx.strokeStyle = litegraph_base.WIDGET_OUTLINE_COLOR |
|
|
ctx.fillStyle = litegraph_base.WIDGET_BGCOLOR |
|
|
ctx.beginPath() |
|
|
if (show_text) |
|
|
ctx.roundRect(margin, y, widget_width - margin * 2, H, [H * 0.5]) |
|
|
else ctx.rect(margin, y, widget_width - margin * 2, H) |
|
|
ctx.fill() |
|
|
if (show_text) { |
|
|
if (!this.disabled) ctx.stroke() |
|
|
const button = button_action(this) |
|
|
if (button != 'None') { |
|
|
ctx.save() |
|
|
if (button.startsWith('No ')) { |
|
|
ctx.fillStyle = litegraph_base.WIDGET_OUTLINE_COLOR |
|
|
ctx.strokeStyle = litegraph_base.WIDGET_OUTLINE_COLOR |
|
|
} else { |
|
|
ctx.fillStyle = litegraph_base.WIDGET_TEXT_COLOR |
|
|
ctx.strokeStyle = litegraph_base.WIDGET_TEXT_COLOR |
|
|
} |
|
|
ctx.beginPath() |
|
|
if (button.endsWith('Reset')) { |
|
|
ctx.arc(widget_width - margin - 26, y + H/2, 4, Math.PI*3/2, Math.PI) |
|
|
ctx.stroke() |
|
|
ctx.beginPath() |
|
|
ctx.moveTo(widget_width - margin - 26, y + H/2 - 1.5) |
|
|
ctx.lineTo(widget_width - margin - 26, y + H/2 - 6.5) |
|
|
ctx.lineTo(widget_width - margin - 30, y + H/2 - 3.5) |
|
|
ctx.fill() |
|
|
} else { |
|
|
ctx.arc(widget_width - margin - 26, y + H/2, 4, Math.PI*2/3, Math.PI*8/3) |
|
|
ctx.moveTo(widget_width - margin - 26 - 8 ** .5, y + H/2 + 8 ** .5) |
|
|
ctx.lineTo(widget_width - margin - 26 + 8 ** .5, y + H/2 - 8 ** .5) |
|
|
ctx.stroke() |
|
|
} |
|
|
ctx.restore() |
|
|
} |
|
|
ctx.fillStyle = litegraph_base.WIDGET_TEXT_COLOR |
|
|
if (!this.disabled) { |
|
|
ctx.beginPath() |
|
|
ctx.moveTo(margin + 16, y + 5) |
|
|
ctx.lineTo(margin + 6, y + H * 0.5) |
|
|
ctx.lineTo(margin + 16, y + H - 5) |
|
|
ctx.fill() |
|
|
ctx.beginPath() |
|
|
ctx.moveTo(widget_width - margin - 16, y + 5) |
|
|
ctx.lineTo(widget_width - margin - 6, y + H * 0.5) |
|
|
ctx.lineTo(widget_width - margin - 16, y + H - 5) |
|
|
ctx.fill() |
|
|
} |
|
|
let freeWidth = widget_width - (40 + margin * 2 + 20) |
|
|
let [valueText, valueWidth] = fitText(ctx, this.displayValue(), freeWidth) |
|
|
freeWidth -= valueWidth |
|
|
|
|
|
ctx.textAlign = 'left' |
|
|
ctx.fillStyle = litegraph_base.WIDGET_SECONDARY_TEXT_COLOR |
|
|
if (freeWidth > 20) { |
|
|
let [name, nameWidth] = fitText(ctx, this.label || this.name, freeWidth) |
|
|
freeWidth -= nameWidth |
|
|
ctx.fillText(name, margin * 2 + 5, y + H * 0.7) |
|
|
} |
|
|
|
|
|
let value_offset = margin * 2 + 20 |
|
|
ctx.textAlign = 'right' |
|
|
if (this.options.unit) { |
|
|
ctx.fillStyle = litegraph_base.WIDGET_OUTLINE_COLOR |
|
|
let [unitText, unitWidth] = fitText(ctx, this.options.unit, freeWidth) |
|
|
if (unitText == this.options.unit) { |
|
|
ctx.fillText(this.options.unit, widget_width - value_offset, y + H * 0.7) |
|
|
value_offset += unitWidth |
|
|
freeWidth -= unitWidth |
|
|
} |
|
|
} |
|
|
ctx.fillStyle = litegraph_base.WIDGET_TEXT_COLOR |
|
|
ctx.fillText(valueText, widget_width - value_offset, y + H * 0.7) |
|
|
ctx.fillStyle = litegraph_base.WIDGET_SECONDARY_TEXT_COLOR |
|
|
|
|
|
|
|
|
let annotation = '' |
|
|
if (this.annotation) { |
|
|
annotation = this.annotation(this.value, freeWidth) |
|
|
} else if ( |
|
|
this.options.annotation && |
|
|
this.value in this.options.annotation |
|
|
) { |
|
|
annotation = this.options.annotation[this.value] |
|
|
} |
|
|
if (annotation) { |
|
|
ctx.fillStyle = litegraph_base.WIDGET_OUTLINE_COLOR |
|
|
let [annoDisplay, annoWidth] = fitText(ctx, annotation, freeWidth) |
|
|
ctx.fillText( |
|
|
annoDisplay, |
|
|
widget_width - 5 - valueWidth - value_offset, |
|
|
y + H * 0.7 |
|
|
) |
|
|
} |
|
|
} |
|
|
} |
|
|
function mouseAnnotated(event, [x, y], node) { |
|
|
|
|
|
|
|
|
|
|
|
const widget_width = this.width || node.size[0] |
|
|
const old_value = this.value |
|
|
const margin = 15 |
|
|
let isButton = 0 |
|
|
if (x > margin + 6 && x < margin + 16) { |
|
|
isButton = -1 |
|
|
} else if (x > widget_width - margin - 16 & x < widget_width - margin - 6) { |
|
|
isButton = 1 |
|
|
} else if (x > widget_width - margin - 34 && x < widget_width - margin - 18) { |
|
|
isButton = 2 |
|
|
} |
|
|
var allow_scroll = true |
|
|
if (allow_scroll && event.type == 'pointermove') { |
|
|
if (event.deltaX) |
|
|
this.value += event.deltaX * (this.options.step || 1) |
|
|
if (this.options.min != null && this.value < this.options.min) { |
|
|
this.value = this.options.min |
|
|
} |
|
|
if (this.options.max != null && this.value > this.options.max) { |
|
|
this.value = this.options.max |
|
|
} |
|
|
} else if (event.type == 'pointerdown') { |
|
|
const buttonType = button_action(this) |
|
|
if (isButton == 2) { |
|
|
if (buttonType == 'Reset') { |
|
|
this.value = this.options.reset |
|
|
} else if (buttonType == 'Disable') { |
|
|
this.value = this.options.disable |
|
|
} |
|
|
} else { |
|
|
this.value += isButton * (this.options.step || 1) |
|
|
if (this.options.min != null && this.value < this.options.min) { |
|
|
this.value = this.options.min |
|
|
} |
|
|
if (this.options.max != null && this.value > this.options.max) { |
|
|
this.value = this.options.max |
|
|
} |
|
|
} |
|
|
} |
|
|
else if (event.type == 'pointerup') { |
|
|
if (event.click_time < 200 && !isButton) { |
|
|
const d_callback = (v) => { |
|
|
this.value = this.parseValue?.(v) ?? Number(v) |
|
|
inner_value_change(this, this.value, node, [x, y]) |
|
|
} |
|
|
const dialog = app.canvas.prompt( |
|
|
'Value', |
|
|
this.value, |
|
|
d_callback, |
|
|
event |
|
|
) |
|
|
const input = dialog.querySelector(".value") |
|
|
input.addEventListener("keydown", (e) => { |
|
|
if (e.keyCode == 9) { |
|
|
e.preventDefault(); |
|
|
e.stopPropagation(); |
|
|
d_callback(input.value) |
|
|
dialog.close() |
|
|
node?.graph?.setDirtyCanvas(true); |
|
|
let i = node.widgets.findIndex((w) => w == this) |
|
|
if (e.shiftKey) |
|
|
i-- |
|
|
else |
|
|
i++ |
|
|
if (node.widgets[i]?.type == "VHS.ANNOTATED") { |
|
|
node.widgets[i]?.mouse(event, [x, y+24], node) |
|
|
} |
|
|
} |
|
|
}) |
|
|
} |
|
|
} |
|
|
|
|
|
if (old_value != this.value) |
|
|
setTimeout( |
|
|
function () { |
|
|
inner_value_change(this, this.value, node, [x, y]) |
|
|
}.bind(this), |
|
|
20 |
|
|
) |
|
|
return true |
|
|
} |
|
|
let latentPreviewNodes = new Set() |
|
|
app.registerExtension({ |
|
|
name: "VideoHelperSuite.Core", |
|
|
settings: [ |
|
|
{ |
|
|
id: 'VHS.AdvancedPreviews', |
|
|
category: ['🎥🅥🅗🅢', 'Previews', 'Advanced Previews'], |
|
|
name: 'Advanced Previews', |
|
|
tooltip: 'Automatically transcode previews on request. Required for advanced functionality', |
|
|
type: 'combo', |
|
|
options: ['Never', 'Always', 'Input Only'], |
|
|
defaultValue: 'Input Only', |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.AdvancedPreviewsMinWidth', |
|
|
category: ['🎥🅥🅗🅢', 'Previews', 'Min Width'], |
|
|
name: 'Minimum preview width', |
|
|
tooltip: 'Advanced previews have their resolution downscaled to the node size for performance. While a node can be resized to increase preview quality, a minimum width can be set that previews won\'t be downscaled beneath. Preveiws will never be upscaled, so this can safely be set large.', |
|
|
type: 'number', |
|
|
attrs: { |
|
|
min: 0, |
|
|
step: 1, |
|
|
max: 3840, |
|
|
}, |
|
|
defaultValue: 0, |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.AdvancedPreviewsDeadline', |
|
|
category: ['🎥🅥🅗🅢', 'Previews', 'Deadline'], |
|
|
name: 'Deadline', |
|
|
tooltip: 'Determines how much time can be spent when encoding advanced previews. Realtime results in reduced quality, but good will likely cause the preview to stutter as initial generation occurs', |
|
|
type: 'combo', |
|
|
options: ['realtime', 'good'], |
|
|
defaultValue: 'realtime', |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.AdvancedPreviewsDefaultMute', |
|
|
category: ['🎥🅥🅗🅢', 'Previews', 'Default Mute'], |
|
|
name: 'Mute videos by default', |
|
|
type: 'boolean', |
|
|
defaultValue: false, |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.LatentPreview', |
|
|
category: ['🎥🅥🅗🅢', 'Sampling', 'Latent Previews'], |
|
|
name: 'Display animated previews when sampling', |
|
|
type: 'boolean', |
|
|
defaultValue: false, |
|
|
onChange(value) { |
|
|
if (!value) { |
|
|
|
|
|
for (let id of latentPreviewNodes) { |
|
|
let n = app.graph.getNodeById(id) |
|
|
let i = n?.widgets?.findIndex((w) => w.name == 'vhslatentpreview') |
|
|
if (i >= 0) { |
|
|
n.widgets.splice(i,1)[0].onRemove() |
|
|
} |
|
|
} |
|
|
latentPreviewNodes = new Set() |
|
|
} |
|
|
}, |
|
|
}, |
|
|
{ |
|
|
id: "VHS.LatentPreviewRate", |
|
|
category: ['🎥🅥🅗🅢', 'Sampling', 'Latent Preview Rate'], |
|
|
name: "Playback rate override.", |
|
|
type: 'number', |
|
|
attrs: { |
|
|
min: 0, |
|
|
step: 1, |
|
|
max: 60 |
|
|
}, |
|
|
tooltip: |
|
|
'Force a specific frame rate for the playback of latent frames. This should not be confused with the output frame rate and will not match for video models.', |
|
|
defaultValue: 0, |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.MetadataImage', |
|
|
category: ['🎥🅥🅗🅢', 'Output', 'MetadataImage'], |
|
|
name: 'Save png of first frame for metadata', |
|
|
type: 'boolean', |
|
|
defaultValue: true, |
|
|
}, |
|
|
{ |
|
|
id: 'VHS.KeepIntermediate', |
|
|
category: ['🎥🅥🅗🅢', 'Output', 'Keep Intermediate'], |
|
|
name: 'Keep required intermediate files after sucessful execution', |
|
|
type: 'boolean', |
|
|
defaultValue: true, |
|
|
}, |
|
|
], |
|
|
|
|
|
async beforeRegisterNodeDef(nodeType, nodeData, app) { |
|
|
if(nodeData?.name?.startsWith("VHS_")) { |
|
|
useKVState(nodeType); |
|
|
if (nodeData.description) { |
|
|
let description = nodeData.description |
|
|
let el = document.createElement("div") |
|
|
el.innerHTML = description |
|
|
if (!el.children.length) { |
|
|
|
|
|
let chunks = description.split('\n') |
|
|
nodeData.description = chunks[0] |
|
|
description = chunks.join('<br>') |
|
|
} else { |
|
|
nodeData.description = el.querySelector('#VHS_shortdesc')?.innerHTML || el.children[1]?.firstChild?.innerHTML |
|
|
} |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function () { |
|
|
helpDOM.addHelp(this, nodeType, description) |
|
|
this.setSize(this.computeSize()) |
|
|
}) |
|
|
} |
|
|
|
|
|
for(let inp of Object.values({...nodeData.input?.required, ...nodeData.input?.optional})) { |
|
|
if (["INT", "FLOAT"].includes(inp[0])) { |
|
|
if (!inp[1]) { |
|
|
inp[1] = {} |
|
|
} |
|
|
inp[1].widgetType ??= "VHS" + inp[0] |
|
|
} |
|
|
} |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function () { |
|
|
let new_widgets = [] |
|
|
if (this.widgets) { |
|
|
for (let w of this.widgets) { |
|
|
let input = this.constructor.nodeData.input |
|
|
let config = input?.required[w.name] ?? input.optional[w.name] |
|
|
if (!config) { |
|
|
continue |
|
|
} |
|
|
if (w?.type == "text" && config[1].vhs_path_extensions) { |
|
|
new_widgets.push(app.widgets.VHSPATH({}, w.name, ["VHSPATH", config[1]])); |
|
|
} else { |
|
|
new_widgets.push(w) |
|
|
} |
|
|
} |
|
|
this.widgets = new_widgets; |
|
|
} |
|
|
const originalAddInput = this.addInput; |
|
|
this.addInput = function(name, type, options) { |
|
|
if (options.widget) { |
|
|
|
|
|
const widget = this.widgets.find((w) => w.name == name) |
|
|
if (widget?.config) { |
|
|
|
|
|
type = widget.config[0] |
|
|
if (type == 'FLOAT') { |
|
|
type = "FLOAT,INT" |
|
|
} |
|
|
setWidgetConfig(options, widget.config) |
|
|
} |
|
|
} |
|
|
return originalAddInput.apply(this, [name, type, options]) |
|
|
} |
|
|
}); |
|
|
} |
|
|
if (nodeData?.name == "VHS_LoadImages") { |
|
|
addUploadWidget(nodeType, nodeData, "directory", "folder"); |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "directory"); |
|
|
chainCallback(pathWidget, "callback", (value) => { |
|
|
if (!value) { |
|
|
return; |
|
|
} |
|
|
let params = {filename : value, type : "input", format: "folder"}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
addLoadCommon(nodeType, nodeData); |
|
|
} else if (nodeData?.name == "VHS_LoadImagesPath") { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "directory"); |
|
|
chainCallback(pathWidget, "callback", (value) => { |
|
|
if (!value) { |
|
|
return; |
|
|
} |
|
|
let params = {filename : value, type : "path", format: "folder"}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
addLoadCommon(nodeType, nodeData); |
|
|
} else if (nodeData?.name == "VHS_LoadVideo" || nodeData?.name == "VHS_LoadVideoFFmpeg") { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "video"); |
|
|
chainCallback(pathWidget, "callback", (value) => { |
|
|
if (!value) { |
|
|
return; |
|
|
} |
|
|
let parts = ["input", value]; |
|
|
let extension_index = parts[1].lastIndexOf("."); |
|
|
let extension = parts[1].slice(extension_index+1); |
|
|
let format = "video" |
|
|
if (["gif", "webp", "avif"].includes(extension)) { |
|
|
format = "image" |
|
|
} |
|
|
format += "/" + extension; |
|
|
let params = {filename : parts[1], type : parts[0], format: format}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
addUploadWidget(nodeType, nodeData, "video"); |
|
|
addLoadCommon(nodeType, nodeData); |
|
|
addVAEOutputToggle(nodeType, nodeData); |
|
|
} else if (nodeData?.name == "VHS_LoadAudio") { |
|
|
addAudioPreview(nodeType) |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "audio_file"); |
|
|
chainCallback(pathWidget, "callback", (filename) => { |
|
|
this.updateParameters({filename, type: 'path'}, true); |
|
|
}); |
|
|
}); |
|
|
} else if (nodeData?.name == "VHS_LoadAudioUpload") { |
|
|
addUploadWidget(nodeType, nodeData, "audio", "audio"); |
|
|
addAudioPreview(nodeType) |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "audio"); |
|
|
chainCallback(pathWidget, "callback", (filename) => { |
|
|
if (!filename) return |
|
|
let params = {filename, type : "input"}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
} else if (nodeData?.name == "VHS_LoadVideoPath" || nodeData?.name == "VHS_LoadVideoFFmpegPath") { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "video"); |
|
|
chainCallback(pathWidget, "callback", (value) => { |
|
|
let extension_index = value.lastIndexOf("."); |
|
|
let extension = value.slice(extension_index+1); |
|
|
let format = "video" |
|
|
if (["gif", "webp", "avif"].includes(extension)) { |
|
|
format = "image" |
|
|
} |
|
|
format += "/" + extension; |
|
|
let params = {filename : value, type: "path", format: format}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
addLoadCommon(nodeType, nodeData); |
|
|
addVAEOutputToggle(nodeType, nodeData); |
|
|
} else if (nodeData?.name == "VHS_LoadImagePath") { |
|
|
addLoadCommon(nodeType, nodeData); |
|
|
addVAEOutputToggle(nodeType, nodeData); |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
const pathWidget = this.widgets.find((w) => w.name === "image"); |
|
|
chainCallback(pathWidget, "callback", (value) => { |
|
|
let extension_index = value.lastIndexOf("."); |
|
|
let extension = value.slice(extension_index+1); |
|
|
let format = "video" + "/" + extension; |
|
|
let params = {filename : value, type: "path", format: format}; |
|
|
this.updateParameters(params, true); |
|
|
}); |
|
|
}); |
|
|
} else if (nodeData?.name == "VHS_VideoCombine") { |
|
|
addDateFormatting(nodeType, "filename_prefix"); |
|
|
chainCallback(nodeType.prototype, "onExecuted", function(message) { |
|
|
if (message?.gifs) { |
|
|
this.updateParameters(message.gifs[0], true); |
|
|
} |
|
|
}); |
|
|
addVideoPreview(nodeType, false); |
|
|
addPreviewOptions(nodeType); |
|
|
addFormatWidgets(nodeType, nodeData); |
|
|
addVAEInputToggle(nodeType, nodeData) |
|
|
} else if (nodeData?.name == "VHS_SaveImageSequence") { |
|
|
|
|
|
|
|
|
|
|
|
} else if (nodeData?.name == "VHS_BatchManager") { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
this.widgets.push({name: "count", type: "dummy", value: 0, |
|
|
computeSize: () => {return [0,-4]}, |
|
|
afterQueued: function() {this.value++;}}); |
|
|
}); |
|
|
} else if (nodeData?.name == "VHS_Unbatch") { |
|
|
cloneType(nodeType, nodeData) |
|
|
} else if (nodeData?.name == "VHS_SelectLatest") { |
|
|
chainCallback(nodeType.prototype, "onNodeCreated", function() { |
|
|
this.isVirtualNode = true |
|
|
chainCallback(this, "onConnectionsChange", function (contype, slot, iscon, linfo) { |
|
|
if (iscon) { |
|
|
this.update_links() |
|
|
} |
|
|
}) |
|
|
|
|
|
this.update_links = function(extraLinks = []) { |
|
|
if (!this.outputs[0].links?.length) return |
|
|
|
|
|
function get_links(node) { |
|
|
let links = [] |
|
|
for (const l of node.outputs[0].links) { |
|
|
const linkInfo = this.graph.links[l] |
|
|
const n = node.graph.getNodeById(linkInfo.target_id) |
|
|
if (n.type == 'Reroute') { |
|
|
links = links.concat(get_links(n)) |
|
|
} else { |
|
|
links.push(l) |
|
|
} |
|
|
} |
|
|
return links |
|
|
} |
|
|
|
|
|
let links = [ |
|
|
...get_links(this).map((l) => this.graph.links[l]), |
|
|
...extraLinks |
|
|
] |
|
|
let v = this.latest_file |
|
|
if (!v) { |
|
|
return |
|
|
} |
|
|
|
|
|
|
|
|
for (const linkInfo of links) { |
|
|
const node = this.graph.getNodeById(linkInfo.target_id) |
|
|
const input = node.inputs[linkInfo.target_slot] |
|
|
const widgetName = input.widget.name |
|
|
const widget = node.widgets.find((w) => w.name === widgetName) |
|
|
if (widget) { |
|
|
widget.value = v |
|
|
if (widget.callback) { |
|
|
widget.callback( widget.value, app.canvas, |
|
|
node, app.canvas.graph_mouse, {}) |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
let fetch_files = async () => { |
|
|
let [path, remainder] = path_stem(this.widgets[0].value) |
|
|
let params = {path : path} |
|
|
let optionsURL = api.apiURL('/vhs/getpath?' + new URLSearchParams(params)); |
|
|
let options |
|
|
try { |
|
|
let resp = await fetch(optionsURL); |
|
|
options = await resp.json(); |
|
|
} catch(e) { |
|
|
options = [] |
|
|
} |
|
|
options = options.filter((file) => file.startsWith(remainder) && file.endsWith(this.widgets[1].value)) |
|
|
if (options.length && this.latest_file != options[options.length-1]) { |
|
|
this.latest_file = path + options[options.length-1] |
|
|
this.update_links() |
|
|
} |
|
|
} |
|
|
this.widgets[0].callback = fetch_files |
|
|
this.widgets[1].callback = fetch_files |
|
|
this.onPromptExecuted = fetch_files |
|
|
this.applyToGraph = this.update_links |
|
|
}) |
|
|
} |
|
|
}, |
|
|
async getCustomWidgets() { |
|
|
return { |
|
|
VHSPATH(node, inputName, inputData) { |
|
|
let w = { |
|
|
name : inputName, |
|
|
type : "VHS.PATH", |
|
|
value : "", |
|
|
draw : function(ctx, node, widget_width, y, H) { |
|
|
|
|
|
var show_text = app.canvas.ds.scale >= (app.canvas.low_quality_zoom_threshold ?? 0.5) |
|
|
var margin = 15; |
|
|
var text_color = LiteGraph.WIDGET_TEXT_COLOR; |
|
|
var secondary_text_color = LiteGraph.WIDGET_SECONDARY_TEXT_COLOR; |
|
|
ctx.textAlign = "left"; |
|
|
ctx.strokeStyle = LiteGraph.WIDGET_OUTLINE_COLOR; |
|
|
ctx.fillStyle = LiteGraph.WIDGET_BGCOLOR; |
|
|
ctx.beginPath(); |
|
|
if (show_text) |
|
|
ctx.roundRect(margin, y, widget_width - margin * 2, H, [H * 0.5]); |
|
|
else |
|
|
ctx.rect( margin, y, widget_width - margin * 2, H ); |
|
|
ctx.fill(); |
|
|
if (show_text) { |
|
|
if(!this.disabled) |
|
|
ctx.stroke(); |
|
|
ctx.save(); |
|
|
ctx.beginPath(); |
|
|
ctx.rect(margin, y, widget_width - margin * 2, H); |
|
|
ctx.clip(); |
|
|
|
|
|
|
|
|
let freeWidth = widget_width - (margin * 2 + 40) |
|
|
ctx.fillStyle = secondary_text_color; |
|
|
const label = this.label || this.name; |
|
|
if (label != null) { |
|
|
let [labelDisplay, labelWidth] = fitText(ctx, label, freeWidth) |
|
|
freeWidth -= labelWidth |
|
|
ctx.fillText(labelDisplay, margin * 2, y + H * 0.7); |
|
|
} |
|
|
ctx.fillStyle = this.value ? text_color : '#777'; |
|
|
ctx.textAlign = "right"; |
|
|
let disp_text = fitPath(ctx, String(this.value || this.options.placeholder), freeWidth)[0] |
|
|
ctx.fillText(disp_text, widget_width - margin * 2, y + H * 0.7); |
|
|
ctx.restore(); |
|
|
} |
|
|
}, |
|
|
mouse : searchBox, |
|
|
options : {}, |
|
|
}; |
|
|
if (inputData.length > 1) { |
|
|
w.options = inputData[1] |
|
|
if (inputData[1].default) { |
|
|
w.value = inputData[1].default; |
|
|
} |
|
|
} |
|
|
|
|
|
if (!node.widgets) { |
|
|
node.widgets = []; |
|
|
} |
|
|
node.widgets.push(w); |
|
|
return w; |
|
|
}, |
|
|
VHSFLOAT(node, inputName, inputData) { |
|
|
let w = { |
|
|
name: inputName, |
|
|
type: "VHS.ANNOTATED", |
|
|
value: inputData[1]?.default ?? 0, |
|
|
draw: drawAnnotated, |
|
|
mouse: mouseAnnotated, |
|
|
computeSize(width) { |
|
|
return [width, 20] |
|
|
}, |
|
|
callback(v) { |
|
|
if (this.options.round) { |
|
|
|
|
|
v = Math.round((v + Number.EPSILON) / |
|
|
this.options.round) * this.options.round |
|
|
} |
|
|
if (this.options.max && v > this.options.max) { |
|
|
v = this.options.max |
|
|
} |
|
|
if (this.options.min && v < this.options.max) { |
|
|
v = this.options.min |
|
|
} |
|
|
this.value = v |
|
|
}, |
|
|
config: inputData, |
|
|
displayValue: function() { |
|
|
return roundToPrecision(this.value, this.options.precision ?? 3) |
|
|
}, |
|
|
options: Object.assign({}, inputData[1]) |
|
|
} |
|
|
if (!node.widgets) { |
|
|
node.widgets = [] |
|
|
} |
|
|
node.widgets.push(w) |
|
|
return w |
|
|
}, |
|
|
VHSINT(node, inputName, inputData) { |
|
|
let w = { |
|
|
name: inputName, |
|
|
type: "VHS.ANNOTATED", |
|
|
value: inputData[1]?.default ?? 0, |
|
|
draw: drawAnnotated, |
|
|
mouse: mouseAnnotated, |
|
|
computeSize(width) { |
|
|
return [width, 20] |
|
|
}, |
|
|
callback(v) { |
|
|
if (this.options.max && v > this.options.max) { |
|
|
v = this.options.max |
|
|
} |
|
|
if (this.options.min && v < this.options.min) { |
|
|
v = this.options.min |
|
|
} |
|
|
if (v == 0) { |
|
|
return |
|
|
} |
|
|
const s = this.options.step |
|
|
let sh = this.options.mod ?? 0 |
|
|
this.value = Math.round((v - sh) / s) * s + sh |
|
|
}, |
|
|
config: inputData, |
|
|
displayValue: function() { |
|
|
return this.value | 0 |
|
|
}, |
|
|
options: Object.assign({}, inputData[1]) |
|
|
} |
|
|
if (!node.widgets) { |
|
|
node.widgets = [] |
|
|
} |
|
|
node.widgets.push(w) |
|
|
return w |
|
|
}, |
|
|
VHSTIMESTAMP(node, inputName, inputData) { |
|
|
let w = { |
|
|
name: inputName, |
|
|
type: "VHS.TIMESTAMP", |
|
|
value: inputData[1]?.default ?? 0, |
|
|
draw: drawAnnotated, |
|
|
mouse: mouseAnnotated, |
|
|
computeSize(width) { |
|
|
return [width, 20] |
|
|
}, |
|
|
parseValue(v) { |
|
|
if (typeof(v) == "string") { |
|
|
let val = 0 |
|
|
for (let chunk of v.split(":")) { |
|
|
val = val * 60 + parseFloat(chunk) |
|
|
} |
|
|
return val |
|
|
} |
|
|
}, |
|
|
callback(v) {}, |
|
|
config: inputData, |
|
|
options: Object.assign({}, inputData[1]), |
|
|
displayValue() { |
|
|
let seconds = this.value |
|
|
let hours = seconds / 3600 | 0 |
|
|
seconds -= 3600 * hours |
|
|
let minutes = seconds / 60 | 0 |
|
|
seconds -= 60 * minutes |
|
|
let display = "" |
|
|
if (hours > 0) { |
|
|
display += hours + ":" |
|
|
} |
|
|
if (hours > 0 || minutes > 0) { |
|
|
if (hours > 0) { |
|
|
minutes = (''+minutes).padStart(2,'0') |
|
|
} |
|
|
display += minutes + ":" |
|
|
} |
|
|
seconds = roundToPrecision(seconds, 4) |
|
|
if ((seconds[1] == '.' || seconds.length == 1) && (minutes > 0 || hours > 0)) { |
|
|
seconds = '0'+seconds |
|
|
} |
|
|
display += seconds |
|
|
return display |
|
|
} |
|
|
} |
|
|
if (!node.widgets) { |
|
|
node.widgets = [] |
|
|
} |
|
|
node.widgets.push(w) |
|
|
return w |
|
|
}, |
|
|
} |
|
|
}, |
|
|
async loadedGraphNode(node) { |
|
|
|
|
|
if (node.type?.startsWith("VHS_") && node.inputs) { |
|
|
const batchInput = node.inputs.find((i) => i.name == "batch_manager") |
|
|
if (batchInput) { |
|
|
batchInput.name = "meta_batch" |
|
|
} |
|
|
} |
|
|
}, |
|
|
async beforeConfigureGraph(graphData, missingNodeTypes) { |
|
|
if(helpDOM?.node) { |
|
|
helpDOM.node = undefined |
|
|
} |
|
|
}, |
|
|
async setup() { |
|
|
let originalGraphToPrompt = app.graphToPrompt |
|
|
let graphToPrompt = async function() { |
|
|
let res = await originalGraphToPrompt.apply(this, arguments); |
|
|
res.workflow.extra['VHS_latentpreview'] = app.ui.settings.getSettingValue("VHS.LatentPreview") |
|
|
res.workflow.extra['VHS_latentpreviewrate'] = app.ui.settings.getSettingValue("VHS.LatentPreviewRate") |
|
|
res.workflow.extra['VHS_MetadataImage'] = app.ui.settings.getSettingValue("VHS.MetadataImage") |
|
|
res.workflow.extra['VHS_KeepIntermediate'] = app.ui.settings.getSettingValue("VHS.KeepIntermediate") |
|
|
return res |
|
|
} |
|
|
app.graphToPrompt = graphToPrompt |
|
|
|
|
|
document.addEventListener('paste', async (e) => { |
|
|
if (!e.target.classList.contains('litegraph') && |
|
|
!e.target.classList.contains('graph-canvas-container')) { |
|
|
return |
|
|
} |
|
|
let data = e.clipboardData || window.clipboardData |
|
|
let filepath = data.getData('text/plain') |
|
|
let video |
|
|
for (const item of data.items) { |
|
|
if (item.type.startsWith('video/')) { |
|
|
video = item |
|
|
break |
|
|
} |
|
|
} |
|
|
if (filepath && copiedPath == filepath) { |
|
|
|
|
|
const pastedNode = LiteGraph.createNode('VHS_LoadVideoPath') |
|
|
app.canvas.graph.add(pastedNode) |
|
|
pastedNode.pos[0] = app.canvas.graph_mouse[0] |
|
|
pastedNode.pos[1] = app.canvas.graph_mouse[1] |
|
|
pastedNode.widgets[0].value = filepath |
|
|
pastedNode.widgets[0].callback?.(filepath) |
|
|
} else if (video && false) { |
|
|
|
|
|
|
|
|
const pastedNode = LiteGraph.createNode('VHS_LoadVideo') |
|
|
app.canvas.graph.add(pastedNode) |
|
|
pastedNode.pos[0] = app.canvas.graph_mouse[0] |
|
|
pastedNode.pos[1] = app.canvas.graph_mouse[1] |
|
|
const pathWidget = pastedNode.widgets[0] |
|
|
|
|
|
const blob = video.getAsFile() |
|
|
const resp = await uploadFile(blob) |
|
|
if (resp.status != 200) { |
|
|
|
|
|
return; |
|
|
} |
|
|
const filename = (await resp.json()).name; |
|
|
pathWidget.options.values.push(filename); |
|
|
pathWidget.value = filename; |
|
|
pathWidget.callback?.(filename) |
|
|
} else { |
|
|
return |
|
|
} |
|
|
e.preventDefault() |
|
|
e.stopImmediatePropagation() |
|
|
return false |
|
|
}, true) |
|
|
}, |
|
|
async init() { |
|
|
if (app.ui.settings.getSettingValue("VHS.AdvancedPreviews") == true) { |
|
|
app.ui.settings.setSettingValue("VHS.AdvancedPreviews", 'Always') |
|
|
} |
|
|
if (app.ui.settings.getSettingValue("VHS.AdvancedPreviews") == false) { |
|
|
app.ui.settings.setSettingValue("VHS.AdvancedPreviews", 'Never') |
|
|
} |
|
|
if (app.VHSHelp != helpDOM) { |
|
|
helpDOM = app.VHSHelp |
|
|
} else { |
|
|
initHelpDOM() |
|
|
} |
|
|
let e = app.extensions.filter((w) => w.name == 'UVR5.AudioPreviewer') |
|
|
if (e.length) { |
|
|
let orig = e[0].beforeRegisterNodeDef |
|
|
e[0].beforeRegisterNodeDef = function(nodeType, nodeData, app) { |
|
|
if(!nodeData?.name?.startsWith("VHS_")) { |
|
|
return orig.apply(this, arguments); |
|
|
} |
|
|
} |
|
|
} |
|
|
}, |
|
|
}); |
|
|
let previewImages = [] |
|
|
api.addEventListener('executing', ({ detail }) => { |
|
|
if (detail === null) { |
|
|
for (let graph of [app.graph, ...app.graph.subgraphs.values()]) { |
|
|
for (let node of graph._nodes) { |
|
|
if (node.type.startsWith("VHS_")) { |
|
|
node.onPromptExecuted?.() |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
}) |
|
|
function getLatentPreviewCtx(id, width, height) { |
|
|
const node = getNodeById(id) |
|
|
if (!node) { |
|
|
return undefined |
|
|
} |
|
|
|
|
|
let previewWidget = node.widgets.find((w) => w.name == "vhslatentpreview") |
|
|
if (!previewWidget) { |
|
|
let canvasEl = document.createElement("canvas") |
|
|
previewWidget = node.addDOMWidget("vhslatentpreview", "vhscanvas", canvasEl, { |
|
|
serialize: false, |
|
|
hideOnZoom: false, |
|
|
}); |
|
|
previewWidget.serialize = false |
|
|
allowDragFromWidget(previewWidget) |
|
|
canvasEl.addEventListener('contextmenu', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousedown_callback(e) |
|
|
}, true); |
|
|
canvasEl.addEventListener('pointerdown', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousedown_callback(e) |
|
|
}, true); |
|
|
canvasEl.addEventListener('mousewheel', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousewheel_callback(e) |
|
|
}, true); |
|
|
canvasEl.addEventListener('pointermove', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mousemove_callback(e) |
|
|
}, true); |
|
|
canvasEl.addEventListener('pointerup', (e) => { |
|
|
e.preventDefault() |
|
|
return app.canvas._mouseup_callback(e) |
|
|
}, true); |
|
|
|
|
|
previewWidget.computeSize = function(width) { |
|
|
if (this.aspectRatio) { |
|
|
let height = (node.size[0]-20)/ this.aspectRatio + 10; |
|
|
if (!(height > 0)) { |
|
|
height = 0; |
|
|
} |
|
|
this.computedHeight = height + 10; |
|
|
return [width, height]; |
|
|
} |
|
|
return [width, -4]; |
|
|
} |
|
|
} |
|
|
let canvasEl = previewWidget.element |
|
|
if (!previewWidget.ctx || canvasEl.width != width |
|
|
|| canvasEl.height != height) { |
|
|
previewWidget.aspectRatio = width / height |
|
|
canvasEl.width = width |
|
|
canvasEl.height = height |
|
|
fitHeight(node) |
|
|
} |
|
|
return canvasEl.getContext("2d") |
|
|
} |
|
|
let animateIntervals = {} |
|
|
function beginLatentPreview(id, previewImages, rate) { |
|
|
latentPreviewNodes.add(id) |
|
|
if (animateIntervals[id]) { |
|
|
clearTimeout(animateIntervals[id]) |
|
|
} |
|
|
let displayIndex = 0 |
|
|
let node = getNodeById(id) |
|
|
|
|
|
|
|
|
node.progress = 0 |
|
|
animateIntervals[id] = setInterval(() => { |
|
|
if (getNodeById(id)?.progress == undefined |
|
|
|| app.canvas.graph.rootGraph != node.graph.rootGraph) { |
|
|
clearTimeout(animateIntervals[id]) |
|
|
delete animateIntervals[id] |
|
|
return |
|
|
} |
|
|
if (!previewImages[displayIndex]) { |
|
|
return |
|
|
} |
|
|
getLatentPreviewCtx(id, previewImages[displayIndex].width, |
|
|
previewImages[displayIndex].height)?.drawImage?.(previewImages[displayIndex],0,0) |
|
|
displayIndex = (displayIndex + 1) % previewImages.length |
|
|
}, 1000/rate); |
|
|
|
|
|
} |
|
|
let previewImagesDict = {} |
|
|
api.addEventListener('VHS_latentpreview', ({ detail }) => { |
|
|
if (detail.id == null) { |
|
|
return |
|
|
} |
|
|
let previewImages = previewImagesDict[detail.id] = [] |
|
|
previewImages.length = detail.length |
|
|
|
|
|
let idParts = detail.id.split(':') |
|
|
for (let i=1; i <= idParts.length; i++) { |
|
|
let id = idParts.slice(0,i).join(':') |
|
|
beginLatentPreview(id, previewImages, detail.rate) |
|
|
} |
|
|
}); |
|
|
let td = new TextDecoder() |
|
|
api.addEventListener('b_preview', async (e) => { |
|
|
if (Object.keys(animateIntervals).length == 0) { |
|
|
return |
|
|
} |
|
|
e.preventDefault() |
|
|
e.stopImmediatePropagation() |
|
|
e.stopPropagation() |
|
|
const dv = new DataView(await e.detail.slice(0,24).arrayBuffer()) |
|
|
const index = dv.getUint32(4) |
|
|
const idlen = dv.getUint8(8) |
|
|
const id = td.decode(dv.buffer.slice(9,9+idlen)) |
|
|
previewImagesDict[id][index] = await window.createImageBitmap(e.detail.slice(24)) |
|
|
return false |
|
|
}, true); |
|
|
|