easydiffusion/ui/media/js/searchable-models.js

664 lines
24 KiB
JavaScript
Raw Normal View History

"use strict"
let modelsCache
let modelsOptions
/*
*** SEARCHABLE MODELS ***
Creates searchable dropdowns for SD, VAE, or HN models.
Also adds a reload models button (placed next to SD models, reloads everything including VAE and HN models).
More reload buttons may be added at strategic UI locations as needed.
Merely calling getModels() makes all the magic happen behind the scene to refresh the dropdowns.
HOW TO CREATE A MODEL DROPDOWN:
1) Create an input element. Make sure to add a data-path property, as this is how model dropdowns are identified in auto-save.js.
<input id="stable_diffusion_model" type="text" spellcheck="false" autocomplete="off" class="model-filter" data-path="" />
2) Just declare one of these for your own dropdown (remember to change the element id, e.g. #stable_diffusion_models to your own input's id).
let stableDiffusionModelField = new ModelDropdown(document.querySelector('#stable_diffusion_model'), 'stable-diffusion')
let vaeModelField = new ModelDropdown(document.querySelector('#vae_model'), 'vae', 'None')
let hypernetworkModelField = new ModelDropdown(document.querySelector('#hypernetwork_model'), 'hypernetwork', 'None')
3) Model dropdowns will be refreshed automatically when the reload models button is invoked.
*/
2023-04-27 19:56:56 +02:00
class ModelDropdown {
modelFilter //= document.querySelector("#model-filter")
modelFilterArrow //= document.querySelector("#model-filter-arrow")
modelList //= document.querySelector("#model-list")
modelResult //= document.querySelector("#model-result")
modelNoResult //= document.querySelector("#model-no-result")
2023-04-27 19:56:56 +02:00
currentSelection //= { elem: undefined, value: '', path: ''}
highlightedModelEntry //= undefined
activeModel //= undefined
inputModels //= undefined
modelKey //= undefined
flatModelList //= []
noneEntry //= ''
modelFilterInitialized //= undefined
/* MIMIC A REGULAR INPUT FIELD */
get parentElement() {
return this.modelFilter.parentElement
}
get parentNode() {
return this.modelFilter.parentNode
}
get value() {
return this.modelFilter.dataset.path
}
set value(path) {
this.modelFilter.dataset.path = path
this.selectEntry(path)
}
get disabled() {
return this.modelFilter.disabled
}
set disabled(state) {
this.modelFilter.disabled = state
if (this.modelFilterArrow) {
2023-04-27 19:56:56 +02:00
this.modelFilterArrow.style.color = state ? "dimgray" : ""
}
}
get modelElements() {
2023-04-27 19:56:56 +02:00
return this.modelList.querySelectorAll(".model-file")
}
addEventListener(type, listener, options) {
return this.modelFilter.addEventListener(type, listener, options)
}
dispatchEvent(event) {
return this.modelFilter.dispatchEvent(event)
}
appendChild(option) {
// do nothing
}
// remember 'this' - http://blog.niftysnippets.org/2008/04/you-must-remember-this.html
bind(f, obj) {
return function() {
return f.apply(obj, arguments)
}
}
2023-04-27 19:56:56 +02:00
/* SEARCHABLE INPUT */
constructor(input, modelKey, noneEntry = "") {
this.modelFilter = input
this.noneEntry = noneEntry
this.modelKey = modelKey
2023-04-27 19:56:56 +02:00
if (modelsOptions !== undefined) {
// reuse models from cache (only useful for plugins, which are loaded after models)
this.inputModels = []
let modelKeys = Array.isArray(this.modelKey) ? this.modelKey : [this.modelKey]
for (let i = 0; i < modelKeys.length; i++) {
let key = modelKeys[i]
2023-06-05 05:30:50 +02:00
let k = Array.isArray(modelsOptions[key]) ? modelsOptions[key] : [modelsOptions[key]]
this.inputModels.push(...k)
}
this.populateModels()
}
2023-04-27 19:56:56 +02:00
document.addEventListener(
"refreshModels",
this.bind(function(e) {
// reload the models
this.inputModels = []
let modelKeys = Array.isArray(this.modelKey) ? this.modelKey : [this.modelKey]
for (let i = 0; i < modelKeys.length; i++) {
let key = modelKeys[i]
2023-06-05 05:30:50 +02:00
let k = Array.isArray(modelsOptions[key]) ? modelsOptions[key] : [modelsOptions[key]]
this.inputModels.push(...k)
}
2023-04-27 19:56:56 +02:00
this.populateModels()
}, this)
)
}
saveCurrentSelection(elem, value, path) {
this.currentSelection.elem = elem
this.currentSelection.value = value
this.currentSelection.path = path
this.modelFilter.dataset.path = path
this.modelFilter.value = value
2023-04-27 19:56:56 +02:00
this.modelFilter.dispatchEvent(new Event("change"))
}
2023-04-27 19:56:56 +02:00
processClick(e) {
e.preventDefault()
2023-04-27 19:56:56 +02:00
if (e.srcElement.classList.contains("model-file") || e.srcElement.classList.contains("fa-file")) {
const elem = e.srcElement.classList.contains("model-file") ? e.srcElement : e.srcElement.parentElement
this.saveCurrentSelection(elem, elem.innerText, elem.dataset.path)
this.hideModelList()
this.modelFilter.focus()
this.modelFilter.select()
}
}
getPreviousVisibleSibling(elem) {
const modelElements = Array.from(this.modelElements)
const index = modelElements.indexOf(elem)
if (index <= 0) {
return undefined
}
2023-04-27 19:56:56 +02:00
return modelElements
.slice(0, index)
.reverse()
.find((e) => e.style.display === "list-item")
}
getLastVisibleChild(elem) {
let lastElementChild = elem.lastElementChild
2023-04-27 19:56:56 +02:00
if (lastElementChild.style.display == "list-item") return lastElementChild
return this.getPreviousVisibleSibling(lastElementChild)
}
2023-04-27 19:56:56 +02:00
getNextVisibleSibling(elem) {
const modelElements = Array.from(this.modelElements)
const index = modelElements.indexOf(elem)
2023-04-27 19:56:56 +02:00
return modelElements.slice(index + 1).find((e) => e.style.display === "list-item")
}
2023-04-27 19:56:56 +02:00
getFirstVisibleChild(elem) {
let firstElementChild = elem.firstElementChild
2023-04-27 19:56:56 +02:00
if (firstElementChild.style.display == "list-item") return firstElementChild
return this.getNextVisibleSibling(firstElementChild)
}
2023-04-27 19:56:56 +02:00
selectModelEntry(elem) {
if (elem) {
if (this.highlightedModelEntry !== undefined) {
2023-04-27 19:56:56 +02:00
this.highlightedModelEntry.classList.remove("selected")
}
this.saveCurrentSelection(elem, elem.innerText, elem.dataset.path)
2023-04-27 19:56:56 +02:00
elem.classList.add("selected")
elem.scrollIntoView({ block: "nearest" })
this.highlightedModelEntry = elem
}
}
2023-04-27 19:56:56 +02:00
selectPreviousFile() {
const elem = this.getPreviousVisibleSibling(this.highlightedModelEntry)
if (elem) {
this.selectModelEntry(elem)
2023-04-27 19:56:56 +02:00
} else {
//this.highlightedModelEntry.parentElement.parentElement.scrollIntoView({block: 'nearest'})
2023-04-27 19:56:56 +02:00
this.highlightedModelEntry.closest(".model-list").scrollTop = 0
}
this.modelFilter.select()
}
2023-04-27 19:56:56 +02:00
selectNextFile() {
this.selectModelEntry(this.getNextVisibleSibling(this.highlightedModelEntry))
this.modelFilter.select()
}
2023-04-27 19:56:56 +02:00
selectFirstFile() {
2023-04-27 19:56:56 +02:00
this.selectModelEntry(this.modelList.querySelector(".model-file"))
this.highlightedModelEntry.scrollIntoView({ block: "nearest" })
this.modelFilter.select()
}
2023-04-27 19:56:56 +02:00
selectLastFile() {
2023-04-27 19:56:56 +02:00
const elems = this.modelList.querySelectorAll(".model-file:last-child")
this.selectModelEntry(elems[elems.length - 1])
this.modelFilter.select()
}
resetSelection() {
this.hideModelList()
this.showAllEntries()
this.modelFilter.value = this.currentSelection.value
this.modelFilter.focus()
this.modelFilter.select()
}
validEntrySelected() {
2023-04-27 19:56:56 +02:00
return this.modelNoResult.style.display === "none"
}
2023-04-27 19:56:56 +02:00
processKey(e) {
switch (e.key) {
2023-04-27 19:56:56 +02:00
case "Escape":
e.preventDefault()
this.resetSelection()
break
2023-04-27 19:56:56 +02:00
case "Enter":
e.preventDefault()
if (this.validEntrySelected()) {
2023-04-27 19:56:56 +02:00
if (this.modelList.style.display != "block") {
this.showModelList()
2023-04-27 19:56:56 +02:00
} else {
this.saveCurrentSelection(
this.highlightedModelEntry,
this.highlightedModelEntry.innerText,
this.highlightedModelEntry.dataset.path
)
this.hideModelList()
this.showAllEntries()
}
this.modelFilter.focus()
2023-04-27 19:56:56 +02:00
} else {
this.resetSelection()
}
break
2023-04-27 19:56:56 +02:00
case "ArrowUp":
e.preventDefault()
if (this.validEntrySelected()) {
this.selectPreviousFile()
}
break
2023-04-27 19:56:56 +02:00
case "ArrowDown":
e.preventDefault()
if (this.validEntrySelected()) {
this.selectNextFile()
}
break
2023-04-27 19:56:56 +02:00
case "ArrowLeft":
if (this.modelList.style.display != "block") {
e.preventDefault()
}
break
2023-04-27 19:56:56 +02:00
case "ArrowRight":
if (this.modelList.style.display != "block") {
e.preventDefault()
}
break
2023-04-27 19:56:56 +02:00
case "PageUp":
e.preventDefault()
if (this.validEntrySelected()) {
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
this.selectPreviousFile()
}
break
2023-04-27 19:56:56 +02:00
case "PageDown":
e.preventDefault()
if (this.validEntrySelected()) {
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
this.selectNextFile()
}
break
2023-04-27 19:56:56 +02:00
case "Home":
//if (this.modelList.style.display != 'block') {
2023-04-27 19:56:56 +02:00
e.preventDefault()
if (this.validEntrySelected()) {
this.selectFirstFile()
}
//}
break
2023-04-27 19:56:56 +02:00
case "End":
//if (this.modelList.style.display != 'block') {
2023-04-27 19:56:56 +02:00
e.preventDefault()
if (this.validEntrySelected()) {
this.selectLastFile()
}
//}
break
default:
2023-04-27 19:56:56 +02:00
//console.log(e.key)
}
}
2023-04-27 19:56:56 +02:00
modelListFocus() {
this.selectEntry()
this.showAllEntries()
}
2023-04-27 19:56:56 +02:00
showModelList() {
2023-04-27 19:56:56 +02:00
this.modelList.style.display = "block"
this.selectEntry()
this.showAllEntries()
//this.modelFilter.value = ''
this.modelFilter.select() // preselect the entire string so user can just start typing.
this.modelFilter.focus()
2023-04-27 19:56:56 +02:00
this.modelFilter.style.cursor = "auto"
}
2023-04-27 19:56:56 +02:00
hideModelList() {
2023-04-27 19:56:56 +02:00
this.modelList.style.display = "none"
this.modelFilter.value = this.currentSelection.value
2023-04-27 19:56:56 +02:00
this.modelFilter.style.cursor = ""
}
2023-04-27 19:56:56 +02:00
toggleModelList(e) {
e.preventDefault()
if (!this.modelFilter.disabled) {
2023-04-27 19:56:56 +02:00
if (this.modelList.style.display != "block") {
this.showModelList()
2023-04-27 19:56:56 +02:00
} else {
this.hideModelList()
this.modelFilter.select()
}
}
}
2023-04-27 19:56:56 +02:00
selectEntry(path) {
if (path !== undefined) {
2023-04-27 19:56:56 +02:00
const entries = this.modelElements
for (const elem of entries) {
if (elem.dataset.path == path) {
this.saveCurrentSelection(elem, elem.innerText, elem.dataset.path)
this.highlightedModelEntry = elem
2023-04-27 19:56:56 +02:00
elem.scrollIntoView({ block: "nearest" })
break
}
}
}
2023-04-27 19:56:56 +02:00
if (this.currentSelection.elem !== undefined) {
// select the previous element
if (this.highlightedModelEntry !== undefined && this.highlightedModelEntry != this.currentSelection.elem) {
2023-04-27 19:56:56 +02:00
this.highlightedModelEntry.classList.remove("selected")
}
2023-04-27 19:56:56 +02:00
this.currentSelection.elem.classList.add("selected")
this.highlightedModelEntry = this.currentSelection.elem
2023-04-27 19:56:56 +02:00
this.currentSelection.elem.scrollIntoView({ block: "nearest" })
} else {
this.selectFirstFile()
}
}
2023-04-27 19:56:56 +02:00
highlightModelAtPosition(e) {
let elem = document.elementFromPoint(e.clientX, e.clientY)
2023-04-27 19:56:56 +02:00
if (elem.classList.contains("model-file")) {
this.highlightModel(elem)
}
}
2023-04-27 19:56:56 +02:00
highlightModel(elem) {
2023-04-27 19:56:56 +02:00
if (elem.classList.contains("model-file")) {
if (this.highlightedModelEntry !== undefined && this.highlightedModelEntry != elem) {
2023-04-27 19:56:56 +02:00
this.highlightedModelEntry.classList.remove("selected")
}
2023-04-27 19:56:56 +02:00
elem.classList.add("selected")
this.highlightedModelEntry = elem
}
}
2023-04-27 19:56:56 +02:00
showAllEntries() {
2023-04-27 19:56:56 +02:00
this.modelList.querySelectorAll("li").forEach(function(li) {
if (li.id !== "model-no-result") {
li.style.display = "list-item"
}
})
2023-04-27 19:56:56 +02:00
this.modelNoResult.style.display = "none"
}
2023-04-27 19:56:56 +02:00
filterList(e) {
const filter = this.modelFilter.value.toLowerCase()
let found = false
let showAllChildren = false
2023-04-27 19:56:56 +02:00
this.modelList.querySelectorAll("li").forEach(function(li) {
if (li.classList.contains("model-folder")) {
showAllChildren = false
}
2023-04-27 19:56:56 +02:00
if (filter == "") {
li.style.display = "list-item"
found = true
} else if (showAllChildren || li.textContent.toLowerCase().match(filter)) {
2023-04-27 19:56:56 +02:00
li.style.display = "list-item"
if (li.classList.contains("model-folder") && li.firstChild.textContent.toLowerCase().match(filter)) {
showAllChildren = true
}
found = true
} else {
2023-04-27 19:56:56 +02:00
li.style.display = "none"
}
})
2023-04-27 19:56:56 +02:00
if (found) {
2023-04-27 19:56:56 +02:00
this.modelResult.style.display = "list-item"
this.modelNoResult.style.display = "none"
const elem = this.getNextVisibleSibling(this.modelList.querySelector(".model-file"))
this.highlightModel(elem)
2023-04-27 19:56:56 +02:00
elem.scrollIntoView({ block: "nearest" })
} else {
this.modelResult.style.display = "none"
this.modelNoResult.style.display = "list-item"
}
2023-04-27 19:56:56 +02:00
this.modelList.style.display = "block"
}
/* MODEL LOADER */
getElementDimensions(element) {
// Clone the element
const clone = element.cloneNode(true)
2023-04-27 19:56:56 +02:00
// Copy the styles of the original element to the cloned element
const originalStyles = window.getComputedStyle(element)
for (let i = 0; i < originalStyles.length; i++) {
const property = originalStyles[i]
clone.style[property] = originalStyles.getPropertyValue(property)
}
2023-04-27 19:56:56 +02:00
// Set its visibility to hidden and display to inline-block
clone.style.visibility = "hidden"
clone.style.display = "inline-block"
2023-04-27 19:56:56 +02:00
// Put the cloned element next to the original element
element.parentNode.insertBefore(clone, element.nextSibling)
2023-04-27 19:56:56 +02:00
// Get its width and height
const width = clone.offsetWidth
const height = clone.offsetHeight
2023-04-27 19:56:56 +02:00
// Remove it from the DOM
clone.remove()
2023-04-27 19:56:56 +02:00
// Return its width and height
return { width, height }
}
2023-04-27 19:56:56 +02:00
/**
2023-04-27 19:56:56 +02:00
* @param {Array<string>} models
*/
sortStringArray(models) {
2023-04-27 19:56:56 +02:00
models.sort((a, b) => a.localeCompare(b, undefined, { sensitivity: "base" }))
}
populateModels() {
this.activeModel = this.modelFilter.dataset.path
2023-04-27 19:56:56 +02:00
this.currentSelection = { elem: undefined, value: "", path: "" }
this.highlightedModelEntry = undefined
this.flatModelList = []
2023-04-27 19:56:56 +02:00
if (this.modelList !== undefined) {
this.modelList.remove()
this.modelFilterArrow.remove()
}
this.createDropdown()
}
createDropdown() {
// create dropdown entries
let rootModelList = this.createRootModelList(this.inputModels)
2023-04-27 19:56:56 +02:00
this.modelFilter.insertAdjacentElement("afterend", rootModelList)
this.modelFilter.insertAdjacentElement(
2023-04-27 19:56:56 +02:00
"afterend",
createElement("i", { id: `${this.modelFilter.id}-model-filter-arrow` }, [
"model-selector-arrow",
"fa-solid",
"fa-angle-down",
2023-04-27 19:56:56 +02:00
])
)
2023-04-27 19:56:56 +02:00
this.modelFilter.classList.add("model-selector")
this.modelFilterArrow = document.querySelector(`#${this.modelFilter.id}-model-filter-arrow`)
if (this.modelFilterArrow) {
2023-04-27 19:56:56 +02:00
this.modelFilterArrow.style.color = this.modelFilter.disabled ? "dimgray" : ""
}
this.modelList = document.querySelector(`#${this.modelFilter.id}-model-list`)
this.modelResult = document.querySelector(`#${this.modelFilter.id}-model-result`)
this.modelNoResult = document.querySelector(`#${this.modelFilter.id}-model-no-result`)
2023-04-27 19:56:56 +02:00
if (this.modelFilterInitialized !== true) {
2023-04-27 19:56:56 +02:00
this.modelFilter.addEventListener("input", this.bind(this.filterList, this))
this.modelFilter.addEventListener("focus", this.bind(this.modelListFocus, this))
this.modelFilter.addEventListener("blur", this.bind(this.hideModelList, this))
this.modelFilter.addEventListener("click", this.bind(this.showModelList, this))
this.modelFilter.addEventListener("keydown", this.bind(this.processKey, this))
this.modelFilterInitialized = true
}
2023-04-27 19:56:56 +02:00
this.modelFilterArrow.addEventListener("mousedown", this.bind(this.toggleModelList, this))
this.modelList.addEventListener("mousemove", this.bind(this.highlightModelAtPosition, this))
this.modelList.addEventListener("mousedown", this.bind(this.processClick, this))
let mf = this.modelFilter
2023-04-27 19:56:56 +02:00
this.modelFilter.addEventListener("focus", function() {
let modelFilterStyle = window.getComputedStyle(mf)
rootModelList.style.minWidth = modelFilterStyle.width
})
this.selectEntry(this.activeModel)
}
/**
* @param {Array<string | object} modelTree
2023-04-27 19:56:56 +02:00
* @param {string} folderName
* @param {boolean} isRootFolder
* @returns {HTMLElement}
*/
createModelNodeList(folderName, modelTree, isRootFolder) {
2023-04-27 19:56:56 +02:00
const listElement = createElement("ul")
const foldersMap = new Map()
const modelsMap = new Map()
2023-04-27 19:56:56 +02:00
modelTree.forEach((model) => {
if (Array.isArray(model)) {
const [childFolderName, childModels] = model
foldersMap.set(
childFolderName,
2023-04-27 19:56:56 +02:00
this.createModelNodeList(`${folderName || ""}/${childFolderName}`, childModels, false)
)
} else {
2023-04-27 19:56:56 +02:00
const classes = ["model-file"]
if (isRootFolder) {
2023-04-27 19:56:56 +02:00
classes.push("in-root-folder")
}
// Remove the leading slash from the model path
const fullPath = folderName ? `${folderName.substring(1)}/${model}` : model
modelsMap.set(
model,
2023-04-27 19:56:56 +02:00
createElement("li", { "data-path": fullPath }, classes, [
createElement("i", undefined, ["fa-regular", "fa-file", "icon"]),
model,
2023-04-27 19:56:56 +02:00
])
)
}
})
const childFolderNames = Array.from(foldersMap.keys())
this.sortStringArray(childFolderNames)
2023-04-27 19:56:56 +02:00
const folderElements = childFolderNames.map((name) => foldersMap.get(name))
const modelNames = Array.from(modelsMap.keys())
this.sortStringArray(modelNames)
2023-04-27 19:56:56 +02:00
const modelElements = modelNames.map((name) => modelsMap.get(name))
if (modelElements.length && folderName) {
2023-03-15 02:43:49 +01:00
listElement.appendChild(
createElement(
2023-04-27 19:56:56 +02:00
"li",
2023-03-15 02:43:49 +01:00
undefined,
2023-04-27 19:56:56 +02:00
["model-folder"],
[createElement("i", undefined, ["fa-regular", "fa-folder-open", "icon"]), folderName.substring(1)]
2023-03-15 02:43:49 +01:00
)
)
}
2023-02-24 14:24:47 +01:00
// const allModelElements = isRootFolder ? [...folderElements, ...modelElements] : [...modelElements, ...folderElements]
const allModelElements = [...modelElements, ...folderElements]
2023-04-27 19:56:56 +02:00
allModelElements.forEach((e) => listElement.appendChild(e))
return listElement
}
/**
* @param {object} modelTree
* @returns {HTMLElement}
*/
createRootModelList(modelTree) {
2023-04-27 19:56:56 +02:00
const rootList = createElement("ul", { id: `${this.modelFilter.id}-model-list` }, ["model-list"])
rootList.appendChild(
2023-04-27 19:56:56 +02:00
createElement("li", { id: `${this.modelFilter.id}-model-no-result` }, ["model-no-result"], "No result")
)
if (this.noneEntry) {
rootList.appendChild(
2023-04-27 19:56:56 +02:00
createElement("li", { "data-path": "" }, ["model-file", "in-root-folder"], this.noneEntry)
)
}
if (modelTree.length > 0) {
2023-04-27 19:56:56 +02:00
const containerListItem = createElement("li", { id: `${this.modelFilter.id}-model-result` }, [
"model-result",
2023-04-27 19:56:56 +02:00
])
//console.log(containerListItem)
containerListItem.appendChild(this.createModelNodeList(undefined, modelTree, true))
rootList.appendChild(containerListItem)
}
return rootList
}
}
/* (RE)LOAD THE MODELS */
async function getModels() {
try {
modelsCache = await SD.getModels()
2023-04-27 19:56:56 +02:00
modelsOptions = modelsCache["options"]
if ("scan-error" in modelsCache) {
// let previewPane = document.getElementById('tab-content-wrapper')
2023-04-27 19:56:56 +02:00
let previewPane = document.getElementById("preview")
previewPane.style.background = "red"
previewPane.style.textAlign = "center"
previewPane.innerHTML =
"<H1>🔥Malware alert!🔥</H1><h2>The file <i>" +
modelsCache["scan-error"] +
'</i> in your <tt>models/stable-diffusion</tt> folder is probably malware infected.</h2><h2>Please delete this file from the folder before proceeding!</h2>After deleting the file, reload this page.<br><br><button onClick="window.location.reload();">Reload Page</button>'
makeImageBtn.disabled = true
}
/* This code should no longer be needed. Commenting out for now, will cleanup later.
const sd_model_setting_key = "stable_diffusion_model"
const vae_model_setting_key = "vae_model"
const hypernetwork_model_key = "hypernetwork_model"
const stableDiffusionOptions = modelsOptions['stable-diffusion']
const vaeOptions = modelsOptions['vae']
const hypernetworkOptions = modelsOptions['hypernetwork']
// TODO: set default for model here too
SETTINGS[sd_model_setting_key].default = stableDiffusionOptions[0]
if (getSetting(sd_model_setting_key) == '' || SETTINGS[sd_model_setting_key].value == '') {
setSetting(sd_model_setting_key, stableDiffusionOptions[0])
}
*/
// notify ModelDropdown objects to refresh
2023-04-27 19:56:56 +02:00
document.dispatchEvent(new Event("refreshModels"))
} catch (e) {
2023-04-27 19:56:56 +02:00
console.log("get models error", e)
}
}
// reload models button
2023-04-27 19:56:56 +02:00
document.querySelector("#reload-models").addEventListener("click", getModels)