|  | import { api } from "./api.js" | 
					
						
						|  | import "./domWidget.js"; | 
					
						
						|  |  | 
					
						
						|  | let controlValueRunBefore = false; | 
					
						
						|  | export function updateControlWidgetLabel(widget) { | 
					
						
						|  | let replacement = "after"; | 
					
						
						|  | let find = "before"; | 
					
						
						|  | if (controlValueRunBefore) { | 
					
						
						|  | [find, replacement] = [replacement, find] | 
					
						
						|  | } | 
					
						
						|  | widget.label = (widget.label ?? widget.name).replace(find, replacement); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | const IS_CONTROL_WIDGET = Symbol(); | 
					
						
						|  | const HAS_EXECUTED = Symbol(); | 
					
						
						|  |  | 
					
						
						|  | function getNumberDefaults(inputData, defaultStep, precision, enable_rounding) { | 
					
						
						|  | let defaultVal = inputData[1]["default"]; | 
					
						
						|  | let { min, max, step, round} = inputData[1]; | 
					
						
						|  |  | 
					
						
						|  | if (defaultVal == undefined) defaultVal = 0; | 
					
						
						|  | if (min == undefined) min = 0; | 
					
						
						|  | if (max == undefined) max = 2048; | 
					
						
						|  | if (step == undefined) step = defaultStep; | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | if (precision == undefined) { | 
					
						
						|  | precision = Math.max(-Math.floor(Math.log10(step)),0); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | if (enable_rounding && (round == undefined || round === true)) { | 
					
						
						|  |  | 
					
						
						|  | round = Math.round(1000000*Math.pow(0.1,precision))/1000000; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return { val: defaultVal, config: { min, max, step: 10.0 * step, round, precision } }; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | export function addValueControlWidget(node, targetWidget, defaultValue = "randomize", values, widgetName, inputData) { | 
					
						
						|  | let name = inputData[1]?.control_after_generate; | 
					
						
						|  | if(typeof name !== "string") { | 
					
						
						|  | name = widgetName; | 
					
						
						|  | } | 
					
						
						|  | const widgets = addValueControlWidgets(node, targetWidget, defaultValue, { | 
					
						
						|  | addFilterList: false, | 
					
						
						|  | controlAfterGenerateName: name | 
					
						
						|  | }, inputData); | 
					
						
						|  | return widgets[0]; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | export function addValueControlWidgets(node, targetWidget, defaultValue = "randomize", options, inputData) { | 
					
						
						|  | if (!defaultValue) defaultValue = "randomize"; | 
					
						
						|  | if (!options) options = {}; | 
					
						
						|  |  | 
					
						
						|  | const getName = (defaultName, optionName) => { | 
					
						
						|  | let name = defaultName; | 
					
						
						|  | if (options[optionName]) { | 
					
						
						|  | name = options[optionName]; | 
					
						
						|  | } else if (typeof inputData?.[1]?.[defaultName] === "string") { | 
					
						
						|  | name = inputData?.[1]?.[defaultName]; | 
					
						
						|  | } else if (inputData?.[1]?.control_prefix) { | 
					
						
						|  | name = inputData?.[1]?.control_prefix + " " + name | 
					
						
						|  | } | 
					
						
						|  | return name; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | const widgets = []; | 
					
						
						|  | const valueControl = node.addWidget( | 
					
						
						|  | "combo", | 
					
						
						|  | getName("control_after_generate", "controlAfterGenerateName"), | 
					
						
						|  | defaultValue, | 
					
						
						|  | function () {}, | 
					
						
						|  | { | 
					
						
						|  | values: ["fixed", "increment", "decrement", "randomize"], | 
					
						
						|  | serialize: false, | 
					
						
						|  | } | 
					
						
						|  | ); | 
					
						
						|  | valueControl[IS_CONTROL_WIDGET] = true; | 
					
						
						|  | updateControlWidgetLabel(valueControl); | 
					
						
						|  | widgets.push(valueControl); | 
					
						
						|  |  | 
					
						
						|  | const isCombo = targetWidget.type === "combo"; | 
					
						
						|  | let comboFilter; | 
					
						
						|  | if (isCombo) { | 
					
						
						|  | valueControl.options.values.push("increment-wrap"); | 
					
						
						|  | } | 
					
						
						|  | if (isCombo && options.addFilterList !== false) { | 
					
						
						|  | comboFilter = node.addWidget( | 
					
						
						|  | "string", | 
					
						
						|  | getName("control_filter_list", "controlFilterListName"), | 
					
						
						|  | "", | 
					
						
						|  | function () {}, | 
					
						
						|  | { | 
					
						
						|  | serialize: false, | 
					
						
						|  | } | 
					
						
						|  | ); | 
					
						
						|  | updateControlWidgetLabel(comboFilter); | 
					
						
						|  |  | 
					
						
						|  | widgets.push(comboFilter); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | const applyWidgetControl = () => { | 
					
						
						|  | var v = valueControl.value; | 
					
						
						|  |  | 
					
						
						|  | if (isCombo && v !== "fixed") { | 
					
						
						|  | let values = targetWidget.options.values; | 
					
						
						|  | const filter = comboFilter?.value; | 
					
						
						|  | if (filter) { | 
					
						
						|  | let check; | 
					
						
						|  | if (filter.startsWith("/") && filter.endsWith("/")) { | 
					
						
						|  | try { | 
					
						
						|  | const regex = new RegExp(filter.substring(1, filter.length - 1)); | 
					
						
						|  | check = (item) => regex.test(item); | 
					
						
						|  | } catch (error) { | 
					
						
						|  | console.error("Error constructing RegExp filter for node " + node.id, filter, error); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | if (!check) { | 
					
						
						|  | const lower = filter.toLocaleLowerCase(); | 
					
						
						|  | check = (item) => item.toLocaleLowerCase().includes(lower); | 
					
						
						|  | } | 
					
						
						|  | values = values.filter(item => check(item)); | 
					
						
						|  | if (!values.length && targetWidget.options.values.length) { | 
					
						
						|  | console.warn("Filter for node " + node.id + " has filtered out all items", filter); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | let current_index = values.indexOf(targetWidget.value); | 
					
						
						|  | let current_length = values.length; | 
					
						
						|  |  | 
					
						
						|  | switch (v) { | 
					
						
						|  | case "increment": | 
					
						
						|  | current_index += 1; | 
					
						
						|  | break; | 
					
						
						|  | case "increment-wrap": | 
					
						
						|  | current_index += 1; | 
					
						
						|  | if ( current_index >= current_length ) { | 
					
						
						|  | current_index = 0; | 
					
						
						|  | } | 
					
						
						|  | break; | 
					
						
						|  | case "decrement": | 
					
						
						|  | current_index -= 1; | 
					
						
						|  | break; | 
					
						
						|  | case "randomize": | 
					
						
						|  | current_index = Math.floor(Math.random() * current_length); | 
					
						
						|  | default: | 
					
						
						|  | break; | 
					
						
						|  | } | 
					
						
						|  | current_index = Math.max(0, current_index); | 
					
						
						|  | current_index = Math.min(current_length - 1, current_index); | 
					
						
						|  | if (current_index >= 0) { | 
					
						
						|  | let value = values[current_index]; | 
					
						
						|  | targetWidget.value = value; | 
					
						
						|  | targetWidget.callback(value); | 
					
						
						|  | } | 
					
						
						|  | } else { | 
					
						
						|  |  | 
					
						
						|  | let min = targetWidget.options.min; | 
					
						
						|  | let max = targetWidget.options.max; | 
					
						
						|  |  | 
					
						
						|  | max = Math.min(1125899906842624, max); | 
					
						
						|  | min = Math.max(-1125899906842624, min); | 
					
						
						|  | let range = (max - min) / (targetWidget.options.step / 10); | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | switch (v) { | 
					
						
						|  | case "fixed": | 
					
						
						|  | break; | 
					
						
						|  | case "increment": | 
					
						
						|  | targetWidget.value += targetWidget.options.step / 10; | 
					
						
						|  | break; | 
					
						
						|  | case "decrement": | 
					
						
						|  | targetWidget.value -= targetWidget.options.step / 10; | 
					
						
						|  | break; | 
					
						
						|  | case "randomize": | 
					
						
						|  | targetWidget.value = Math.floor(Math.random() * range) * (targetWidget.options.step / 10) + min; | 
					
						
						|  | default: | 
					
						
						|  | break; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | if (targetWidget.value < min) targetWidget.value = min; | 
					
						
						|  |  | 
					
						
						|  | if (targetWidget.value > max) | 
					
						
						|  | targetWidget.value = max; | 
					
						
						|  | targetWidget.callback(targetWidget.value); | 
					
						
						|  | } | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  | valueControl.beforeQueued = () => { | 
					
						
						|  | if (controlValueRunBefore) { | 
					
						
						|  |  | 
					
						
						|  | if (valueControl[HAS_EXECUTED]) { | 
					
						
						|  | applyWidgetControl(); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | valueControl[HAS_EXECUTED] = true; | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  | valueControl.afterQueued = () => { | 
					
						
						|  | if (!controlValueRunBefore) { | 
					
						
						|  | applyWidgetControl(); | 
					
						
						|  | } | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  | return widgets; | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  | function seedWidget(node, inputName, inputData, app, widgetName) { | 
					
						
						|  | const seed = createIntWidget(node, inputName, inputData, app, true); | 
					
						
						|  | const seedControl = addValueControlWidget(node, seed.widget, "randomize", undefined, widgetName, inputData); | 
					
						
						|  |  | 
					
						
						|  | seed.widget.linkedWidgets = [seedControl]; | 
					
						
						|  | return seed; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | function createIntWidget(node, inputName, inputData, app, isSeedInput) { | 
					
						
						|  | const control = inputData[1]?.control_after_generate; | 
					
						
						|  | if (!isSeedInput && control) { | 
					
						
						|  | return seedWidget(node, inputName, inputData, app, typeof control === "string" ? control : undefined); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | let widgetType = isSlider(inputData[1]["display"], app); | 
					
						
						|  | const { val, config } = getNumberDefaults(inputData, 1, 0, true); | 
					
						
						|  | Object.assign(config, { precision: 0 }); | 
					
						
						|  | return { | 
					
						
						|  | widget: node.addWidget( | 
					
						
						|  | widgetType, | 
					
						
						|  | inputName, | 
					
						
						|  | val, | 
					
						
						|  | function (v) { | 
					
						
						|  | const s = this.options.step / 10; | 
					
						
						|  | this.value = Math.round(v / s) * s; | 
					
						
						|  | }, | 
					
						
						|  | config | 
					
						
						|  | ), | 
					
						
						|  | }; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | function addMultilineWidget(node, name, opts, app) { | 
					
						
						|  | const inputEl = document.createElement("textarea"); | 
					
						
						|  | inputEl.className = "comfy-multiline-input"; | 
					
						
						|  | inputEl.value = opts.defaultVal; | 
					
						
						|  | inputEl.placeholder = opts.placeholder || name; | 
					
						
						|  |  | 
					
						
						|  | const widget = node.addDOMWidget(name, "customtext", inputEl, { | 
					
						
						|  | getValue() { | 
					
						
						|  | return inputEl.value; | 
					
						
						|  | }, | 
					
						
						|  | setValue(v) { | 
					
						
						|  | inputEl.value = v; | 
					
						
						|  | }, | 
					
						
						|  | }); | 
					
						
						|  | widget.inputEl = inputEl; | 
					
						
						|  |  | 
					
						
						|  | inputEl.addEventListener("input", () => { | 
					
						
						|  | widget.callback?.(widget.value); | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | return { minWidth: 400, minHeight: 200, widget }; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | function isSlider(display, app) { | 
					
						
						|  | if (app.ui.settings.getSettingValue("Comfy.DisableSliders")) { | 
					
						
						|  | return "number" | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return (display==="slider") ? "slider" : "number" | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | export function initWidgets(app) { | 
					
						
						|  | app.ui.settings.addSetting({ | 
					
						
						|  | id: "Comfy.WidgetControlMode", | 
					
						
						|  | name: "Widget Value Control Mode", | 
					
						
						|  | type: "combo", | 
					
						
						|  | defaultValue: "after", | 
					
						
						|  | options: ["before", "after"], | 
					
						
						|  | tooltip: "Controls when widget values are updated (randomize/increment/decrement), either before the prompt is queued or after.", | 
					
						
						|  | onChange(value) { | 
					
						
						|  | controlValueRunBefore = value === "before"; | 
					
						
						|  | for (const n of app.graph._nodes) { | 
					
						
						|  | if (!n.widgets) continue; | 
					
						
						|  | for (const w of n.widgets) { | 
					
						
						|  | if (w[IS_CONTROL_WIDGET]) { | 
					
						
						|  | updateControlWidgetLabel(w); | 
					
						
						|  | if (w.linkedWidgets) { | 
					
						
						|  | for (const l of w.linkedWidgets) { | 
					
						
						|  | updateControlWidgetLabel(l); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | app.graph.setDirtyCanvas(true); | 
					
						
						|  | }, | 
					
						
						|  | }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | export const ComfyWidgets = { | 
					
						
						|  | "INT:seed": seedWidget, | 
					
						
						|  | "INT:noise_seed": seedWidget, | 
					
						
						|  | FLOAT(node, inputName, inputData, app) { | 
					
						
						|  | let widgetType = isSlider(inputData[1]["display"], app); | 
					
						
						|  | let precision = app.ui.settings.getSettingValue("Comfy.FloatRoundingPrecision"); | 
					
						
						|  | let disable_rounding = app.ui.settings.getSettingValue("Comfy.DisableFloatRounding") | 
					
						
						|  | if (precision == 0) precision = undefined; | 
					
						
						|  | const { val, config } = getNumberDefaults(inputData, 0.5, precision, !disable_rounding); | 
					
						
						|  | return { widget: node.addWidget(widgetType, inputName, val, | 
					
						
						|  | function (v) { | 
					
						
						|  | if (config.round) { | 
					
						
						|  | this.value = Math.round(v/config.round)*config.round; | 
					
						
						|  | } else { | 
					
						
						|  | this.value = v; | 
					
						
						|  | } | 
					
						
						|  | }, config) }; | 
					
						
						|  | }, | 
					
						
						|  | INT(node, inputName, inputData, app) { | 
					
						
						|  | return createIntWidget(node, inputName, inputData, app); | 
					
						
						|  | }, | 
					
						
						|  | BOOLEAN(node, inputName, inputData) { | 
					
						
						|  | let defaultVal = false; | 
					
						
						|  | let options = {}; | 
					
						
						|  | if (inputData[1]) { | 
					
						
						|  | if (inputData[1].default) | 
					
						
						|  | defaultVal = inputData[1].default; | 
					
						
						|  | if (inputData[1].label_on) | 
					
						
						|  | options["on"] = inputData[1].label_on; | 
					
						
						|  | if (inputData[1].label_off) | 
					
						
						|  | options["off"] = inputData[1].label_off; | 
					
						
						|  | } | 
					
						
						|  | return { | 
					
						
						|  | widget: node.addWidget( | 
					
						
						|  | "toggle", | 
					
						
						|  | inputName, | 
					
						
						|  | defaultVal, | 
					
						
						|  | () => {}, | 
					
						
						|  | options, | 
					
						
						|  | ) | 
					
						
						|  | }; | 
					
						
						|  | }, | 
					
						
						|  | STRING(node, inputName, inputData, app) { | 
					
						
						|  | const defaultVal = inputData[1].default || ""; | 
					
						
						|  | const multiline = !!inputData[1].multiline; | 
					
						
						|  |  | 
					
						
						|  | let res; | 
					
						
						|  | if (multiline) { | 
					
						
						|  | res = addMultilineWidget(node, inputName, { defaultVal, ...inputData[1] }, app); | 
					
						
						|  | } else { | 
					
						
						|  | res = { widget: node.addWidget("text", inputName, defaultVal, () => {}, {}) }; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | if(inputData[1].dynamicPrompts != undefined) | 
					
						
						|  | res.widget.dynamicPrompts = inputData[1].dynamicPrompts; | 
					
						
						|  |  | 
					
						
						|  | return res; | 
					
						
						|  | }, | 
					
						
						|  | COMBO(node, inputName, inputData) { | 
					
						
						|  | const type = inputData[0]; | 
					
						
						|  | let defaultValue = type[0]; | 
					
						
						|  | if (inputData[1] && inputData[1].default) { | 
					
						
						|  | defaultValue = inputData[1].default; | 
					
						
						|  | } | 
					
						
						|  | const res = { widget: node.addWidget("combo", inputName, defaultValue, () => {}, { values: type }) }; | 
					
						
						|  | if (inputData[1]?.control_after_generate) { | 
					
						
						|  | res.widget.linkedWidgets = addValueControlWidgets(node, res.widget, undefined, undefined, inputData); | 
					
						
						|  | } | 
					
						
						|  | return res; | 
					
						
						|  | }, | 
					
						
						|  | IMAGEUPLOAD(node, inputName, inputData, app) { | 
					
						
						|  | const imageWidget = node.widgets.find((w) => w.name === (inputData[1]?.widget ?? "image")); | 
					
						
						|  | let uploadWidget; | 
					
						
						|  |  | 
					
						
						|  | function showImage(name) { | 
					
						
						|  | const img = new Image(); | 
					
						
						|  | img.onload = () => { | 
					
						
						|  | node.imgs = [img]; | 
					
						
						|  | app.graph.setDirtyCanvas(true); | 
					
						
						|  | }; | 
					
						
						|  | let folder_separator = name.lastIndexOf("/"); | 
					
						
						|  | let subfolder = ""; | 
					
						
						|  | if (folder_separator > -1) { | 
					
						
						|  | subfolder = name.substring(0, folder_separator); | 
					
						
						|  | name = name.substring(folder_separator + 1); | 
					
						
						|  | } | 
					
						
						|  | img.src = api.apiURL(`/view?filename=${encodeURIComponent(name)}&type=input&subfolder=${subfolder}${app.getPreviewFormatParam()}${app.getRandParam()}`); | 
					
						
						|  | node.setSizeForImage?.(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | var default_value = imageWidget.value; | 
					
						
						|  | Object.defineProperty(imageWidget, "value", { | 
					
						
						|  | set : function(value) { | 
					
						
						|  | this._real_value = value; | 
					
						
						|  | }, | 
					
						
						|  |  | 
					
						
						|  | get : function() { | 
					
						
						|  | let value = ""; | 
					
						
						|  | if (this._real_value) { | 
					
						
						|  | value = this._real_value; | 
					
						
						|  | } else { | 
					
						
						|  | return default_value; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | if (value.filename) { | 
					
						
						|  | let real_value = value; | 
					
						
						|  | value = ""; | 
					
						
						|  | if (real_value.subfolder) { | 
					
						
						|  | value = real_value.subfolder + "/"; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | value += real_value.filename; | 
					
						
						|  |  | 
					
						
						|  | if(real_value.type && real_value.type !== "input") | 
					
						
						|  | value += ` [${real_value.type}]`; | 
					
						
						|  | } | 
					
						
						|  | return value; | 
					
						
						|  | } | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | const cb = node.callback; | 
					
						
						|  | imageWidget.callback = function () { | 
					
						
						|  | showImage(imageWidget.value); | 
					
						
						|  | if (cb) { | 
					
						
						|  | return cb.apply(this, arguments); | 
					
						
						|  | } | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | requestAnimationFrame(() => { | 
					
						
						|  | if (imageWidget.value) { | 
					
						
						|  | showImage(imageWidget.value); | 
					
						
						|  | } | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | async function uploadFile(file, updateNode, pasted = false) { | 
					
						
						|  | try { | 
					
						
						|  |  | 
					
						
						|  | const body = new FormData(); | 
					
						
						|  | body.append("image", file); | 
					
						
						|  | if (pasted) body.append("subfolder", "pasted"); | 
					
						
						|  | const resp = await api.fetchApi("/upload/image", { | 
					
						
						|  | method: "POST", | 
					
						
						|  | body, | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | if (resp.status === 200) { | 
					
						
						|  | const data = await resp.json(); | 
					
						
						|  |  | 
					
						
						|  | let path = data.name; | 
					
						
						|  | if (data.subfolder) path = data.subfolder + "/" + path; | 
					
						
						|  |  | 
					
						
						|  | if (!imageWidget.options.values.includes(path)) { | 
					
						
						|  | imageWidget.options.values.push(path); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | if (updateNode) { | 
					
						
						|  | showImage(path); | 
					
						
						|  | imageWidget.value = path; | 
					
						
						|  | } | 
					
						
						|  | } else { | 
					
						
						|  | alert(resp.status + " - " + resp.statusText); | 
					
						
						|  | } | 
					
						
						|  | } catch (error) { | 
					
						
						|  | alert(error); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | const fileInput = document.createElement("input"); | 
					
						
						|  | Object.assign(fileInput, { | 
					
						
						|  | type: "file", | 
					
						
						|  | accept: "image/jpeg,image/png,image/webp", | 
					
						
						|  | style: "display: none", | 
					
						
						|  | onchange: async () => { | 
					
						
						|  | if (fileInput.files.length) { | 
					
						
						|  | await uploadFile(fileInput.files[0], true); | 
					
						
						|  | } | 
					
						
						|  | }, | 
					
						
						|  | }); | 
					
						
						|  | document.body.append(fileInput); | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | uploadWidget = node.addWidget("button", inputName, "image", () => { | 
					
						
						|  | fileInput.click(); | 
					
						
						|  | }); | 
					
						
						|  | uploadWidget.label = "choose file to upload"; | 
					
						
						|  | uploadWidget.serialize = false; | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | node.onDragOver = function (e) { | 
					
						
						|  | if (e.dataTransfer && e.dataTransfer.items) { | 
					
						
						|  | const image = [...e.dataTransfer.items].find((f) => f.kind === "file"); | 
					
						
						|  | return !!image; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return false; | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | node.onDragDrop = function (e) { | 
					
						
						|  | console.log("onDragDrop called"); | 
					
						
						|  | let handled = false; | 
					
						
						|  | for (const file of e.dataTransfer.files) { | 
					
						
						|  | if (file.type.startsWith("image/")) { | 
					
						
						|  | uploadFile(file, !handled); | 
					
						
						|  | handled = true; | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return handled; | 
					
						
						|  | }; | 
					
						
						|  |  | 
					
						
						|  | node.pasteFile = function(file) { | 
					
						
						|  | if (file.type.startsWith("image/")) { | 
					
						
						|  | const is_pasted = (file.name === "image.png") && | 
					
						
						|  | (file.lastModified - Date.now() < 2000); | 
					
						
						|  | uploadFile(file, true, is_pasted); | 
					
						
						|  | return true; | 
					
						
						|  | } | 
					
						
						|  | return false; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return { widget: uploadWidget }; | 
					
						
						|  | }, | 
					
						
						|  | }; | 
					
						
						|  |  |