easydiffusion/ui/media/js/image-modifiers.js

411 lines
15 KiB
JavaScript
Raw Normal View History

let activeTags = []
let modifiers = []
2022-10-21 10:38:16 +02:00
let customModifiersGroupElement = undefined
let customModifiersInitialContent
2023-04-27 19:56:56 +02:00
let editorModifierEntries = document.querySelector("#editor-modifiers-entries")
let editorModifierTagsList = document.querySelector("#editor-inputs-tags-list")
let editorTagsContainer = document.querySelector("#editor-inputs-tags-container")
let modifierCardSizeSlider = document.querySelector("#modifier-card-size-slider")
let previewImageField = document.querySelector("#preview-image")
let modifierSettingsBtn = document.querySelector("#modifier-settings-btn")
let modifierSettingsOverlay = document.querySelector("#modifier-settings-config")
let customModifiersTextBox = document.querySelector("#custom-modifiers-input")
let customModifierEntriesToolbar = document.querySelector("#editor-modifiers-entries-toolbar")
const modifierThumbnailPath = "media/modifier-thumbnails"
const activeCardClass = "modifier-card-active"
2022-10-21 10:38:16 +02:00
const CUSTOM_MODIFIERS_KEY = "customModifiers"
function createModifierCard(name, previews, removeBy) {
2023-04-27 19:56:56 +02:00
const modifierCard = document.createElement("div")
2023-03-10 21:23:16 +01:00
let style = previewImageField.value
2023-04-27 19:56:56 +02:00
let styleIndex = style == "portrait" ? 0 : 1
2023-03-10 21:23:16 +01:00
2023-04-27 19:56:56 +02:00
modifierCard.className = "modifier-card"
modifierCard.innerHTML = `
<div class="modifier-card-overlay"></div>
<div class="modifier-card-image-container">
<div class="modifier-card-image-overlay">+</div>
<p class="modifier-card-error-label"></p>
<img onerror="this.remove()" alt="Modifier Image" class="modifier-card-image">
</div>
<div class="modifier-card-container">
<div class="modifier-card-label"><p></p></div>
</div>`
2023-04-27 19:56:56 +02:00
const image = modifierCard.querySelector(".modifier-card-image")
const errorText = modifierCard.querySelector(".modifier-card-error-label")
const label = modifierCard.querySelector(".modifier-card-label")
2023-04-27 19:56:56 +02:00
errorText.innerText = "No Image"
2023-04-27 19:56:56 +02:00
if (typeof previews == "object") {
image.src = previews[styleIndex] // portrait
image.setAttribute("preview-type", style)
} else {
image.remove()
}
const maxLabelLength = 30
2023-04-27 19:56:56 +02:00
const cardLabel = removeBy ? name.replace("by ", "") : name
2023-04-27 19:56:56 +02:00
if (cardLabel.length <= maxLabelLength) {
label.querySelector("p").innerText = cardLabel
} else {
2023-04-27 19:56:56 +02:00
const tooltipText = document.createElement("span")
tooltipText.className = "tooltip-text"
tooltipText.innerText = name
2023-04-27 19:56:56 +02:00
label.classList.add("tooltip")
label.appendChild(tooltipText)
2023-04-27 19:56:56 +02:00
label.querySelector("p").innerText = cardLabel.substring(0, maxLabelLength) + "..."
}
2023-04-27 19:56:56 +02:00
label.querySelector("p").dataset.fullName = name // preserve the full name
return modifierCard
}
function createModifierGroup(modifierGroup, initiallyExpanded, removeBy) {
2022-10-21 10:38:16 +02:00
const title = modifierGroup.category
const modifiers = modifierGroup.modifiers
2023-04-27 19:56:56 +02:00
const titleEl = document.createElement("h5")
titleEl.className = "collapsible"
2022-10-21 10:38:16 +02:00
titleEl.innerText = title
2023-04-27 19:56:56 +02:00
const modifiersEl = document.createElement("div")
modifiersEl.classList.add("collapsible-content", "editor-modifiers-leaf")
2022-10-21 10:38:16 +02:00
if (initiallyExpanded === true) {
2023-04-27 19:56:56 +02:00
titleEl.className += " active"
2022-10-21 10:38:16 +02:00
}
2023-04-27 19:56:56 +02:00
modifiers.forEach((modObj) => {
2022-10-21 10:38:16 +02:00
const modifierName = modObj.modifier
2023-04-27 19:56:56 +02:00
const modifierPreviews = modObj?.previews?.map(
(preview) =>
`${IMAGE_REGEX.test(preview.image) ? preview.image : modifierThumbnailPath + "/" + preview.path}`
)
2022-10-21 10:38:16 +02:00
const modifierCard = createModifierCard(modifierName, modifierPreviews, removeBy)
2022-10-21 10:38:16 +02:00
2023-04-27 19:56:56 +02:00
if (typeof modifierCard == "object") {
2022-10-21 10:38:16 +02:00
modifiersEl.appendChild(modifierCard)
2022-11-30 12:02:43 +01:00
const trimmedName = trimModifiers(modifierName)
2022-10-21 10:38:16 +02:00
2023-04-27 19:56:56 +02:00
modifierCard.addEventListener("click", () => {
if (activeTags.map((x) => trimModifiers(x.name)).includes(trimmedName)) {
2022-10-21 10:38:16 +02:00
// remove modifier from active array
2023-04-27 19:56:56 +02:00
activeTags = activeTags.filter((x) => trimModifiers(x.name) != trimmedName)
toggleCardState(trimmedName, false)
2022-10-21 10:38:16 +02:00
} else {
// add modifier to active array
activeTags.push({
2023-04-27 19:56:56 +02:00
name: modifierName,
element: modifierCard.cloneNode(true),
originElement: modifierCard,
previews: modifierPreviews
2022-10-21 10:38:16 +02:00
})
toggleCardState(trimmedName, true)
2022-10-21 10:38:16 +02:00
}
refreshTagsList()
2023-04-27 19:56:56 +02:00
document.dispatchEvent(new Event("refreshImageModifiers"))
2022-10-21 10:38:16 +02:00
})
}
})
2023-04-27 19:56:56 +02:00
let brk = document.createElement("br")
brk.style.clear = "both"
2022-10-21 10:38:16 +02:00
modifiersEl.appendChild(brk)
2023-04-27 19:56:56 +02:00
let e = document.createElement("div")
e.className = "modifier-category"
2022-10-21 10:38:16 +02:00
e.appendChild(titleEl)
e.appendChild(modifiersEl)
editorModifierEntries.insertBefore(e, customModifierEntriesToolbar.nextSibling)
2022-10-21 10:38:16 +02:00
return e
}
function trimModifiers(tag) {
// Remove trailing '-' and/or '+'
2023-04-27 19:56:56 +02:00
tag = tag.replace(/[-+]+$/, "")
// Remove parentheses at beginning and end
2023-04-27 19:56:56 +02:00
return tag.replace(/^[(]+|[\s)]+$/g, "")
}
async function loadModifiers() {
try {
2023-04-27 19:56:56 +02:00
let res = await fetch("/get/modifiers")
if (res.status === 200) {
res = await res.json()
2023-04-27 19:56:56 +02:00
modifiers = res // update global variable
2022-10-21 10:38:16 +02:00
res.reverse()
2022-10-21 10:38:16 +02:00
res.forEach((modifierGroup, idx) => {
2023-04-27 19:56:56 +02:00
createModifierGroup(modifierGroup, idx === res.length - 1, modifierGroup === "Artist" ? true : false) // only remove "By " for artists
})
createCollapsibles(editorModifierEntries)
}
} catch (e) {
2023-04-27 19:56:56 +02:00
console.error("error fetching modifiers", e)
}
2022-10-21 10:38:16 +02:00
loadCustomModifiers()
resizeModifierCards(modifierCardSizeSlider.value)
2023-04-27 19:56:56 +02:00
document.dispatchEvent(new Event("loadImageModifiers"))
}
function refreshModifiersState(newTags, inactiveTags) {
2022-11-11 03:36:39 +01:00
// clear existing modifiers
2023-04-27 19:56:56 +02:00
document
.querySelector("#editor-modifiers")
.querySelectorAll(".modifier-card")
.forEach((modifierCard) => {
const modifierName = modifierCard.querySelector(".modifier-card-label p").dataset.fullName // pick the full modifier name
if (activeTags.map((x) => x.name).includes(modifierName)) {
modifierCard.classList.remove(activeCardClass)
modifierCard.querySelector(".modifier-card-image-overlay").innerText = "+"
}
})
2022-11-11 03:36:39 +01:00
activeTags = []
// set new modifiers
2023-04-27 19:56:56 +02:00
newTags.forEach((tag) => {
2022-11-11 03:36:39 +01:00
let found = false
2023-04-27 19:56:56 +02:00
document
.querySelector("#editor-modifiers")
.querySelectorAll(".modifier-card")
.forEach((modifierCard) => {
const modifierName = modifierCard.querySelector(".modifier-card-label p").dataset.fullName
const shortModifierName = modifierCard.querySelector(".modifier-card-label p").innerText
if (trimModifiers(tag) == trimModifiers(modifierName)) {
// add modifier to active array
if (!activeTags.map((x) => x.name).includes(tag)) {
// only add each tag once even if several custom modifier cards share the same tag
const imageModifierCard = modifierCard.cloneNode(true)
imageModifierCard.querySelector(".modifier-card-label p").innerText = tag.replace(
modifierName,
shortModifierName
)
activeTags.push({
name: tag,
element: imageModifierCard,
originElement: modifierCard
})
}
modifierCard.classList.add(activeCardClass)
modifierCard.querySelector(".modifier-card-image-overlay").innerText = "-"
found = true
}
2023-04-27 19:56:56 +02:00
})
if (found == false) {
// custom tag went missing, create one here
let modifierCard = createModifierCard(tag, undefined, false) // create a modifier card for the missing tag, no image
2023-04-27 19:56:56 +02:00
modifierCard.addEventListener("click", () => {
if (activeTags.map((x) => x.name).includes(tag)) {
2022-11-11 03:36:39 +01:00
// remove modifier from active array
2023-04-27 19:56:56 +02:00
activeTags = activeTags.filter((x) => x.name != tag)
2022-11-11 03:36:39 +01:00
modifierCard.classList.remove(activeCardClass)
2023-04-27 19:56:56 +02:00
modifierCard.querySelector(".modifier-card-image-overlay").innerText = "+"
2022-11-11 03:36:39 +01:00
}
refreshTagsList()
})
activeTags.push({
2023-04-27 19:56:56 +02:00
name: tag,
element: modifierCard,
originElement: undefined // no origin element for missing tags
2022-11-11 03:36:39 +01:00
})
}
})
refreshTagsList(inactiveTags)
2022-11-11 03:36:39 +01:00
}
function refreshInactiveTags(inactiveTags) {
// update inactive tags
if (inactiveTags !== undefined && inactiveTags.length > 0) {
2023-04-27 19:56:56 +02:00
activeTags.forEach((tag) => {
if (inactiveTags.find((element) => element === tag.name) !== undefined) {
tag.inactive = true
}
})
}
2023-04-27 19:56:56 +02:00
// update cards
2023-04-27 19:56:56 +02:00
let overlays = document.querySelector("#editor-inputs-tags-list").querySelectorAll(".modifier-card-overlay")
overlays.forEach((i) => {
let modifierName = i.parentElement.getElementsByClassName("modifier-card-label")[0].getElementsByTagName("p")[0]
.dataset.fullName
if (inactiveTags?.find((element) => element === modifierName) !== undefined) {
i.parentElement.classList.add("modifier-toggle-inactive")
}
})
}
function refreshTagsList(inactiveTags) {
2023-04-27 19:56:56 +02:00
editorModifierTagsList.innerHTML = ""
if (activeTags.length == 0) {
2023-04-27 19:56:56 +02:00
editorTagsContainer.style.display = "none"
return
} else {
2023-04-27 19:56:56 +02:00
editorTagsContainer.style.display = "block"
}
activeTags.forEach((tag, index) => {
2023-04-27 19:56:56 +02:00
tag.element.querySelector(".modifier-card-image-overlay").innerText = "-"
tag.element.classList.add("modifier-card-tiny")
editorModifierTagsList.appendChild(tag.element)
2023-04-27 19:56:56 +02:00
tag.element.addEventListener("click", () => {
let idx = activeTags.findIndex((o) => {
return o.name === tag.name
})
if (idx !== -1) {
toggleCardState(activeTags[idx].name, false)
activeTags.splice(idx, 1)
refreshTagsList()
}
2023-04-27 19:56:56 +02:00
document.dispatchEvent(new Event("refreshImageModifiers"))
})
})
2023-04-27 19:56:56 +02:00
let brk = document.createElement("br")
brk.style.clear = "both"
editorModifierTagsList.appendChild(brk)
refreshInactiveTags(inactiveTags)
2023-04-27 19:56:56 +02:00
document.dispatchEvent(new Event("refreshImageModifiers")) // notify plugins that the image tags have been refreshed
}
function toggleCardState(modifierName, makeActive) {
2023-04-27 19:56:56 +02:00
document
.querySelector("#editor-modifiers")
.querySelectorAll(".modifier-card")
.forEach((card) => {
const name = card.querySelector(".modifier-card-label").innerText
if (
trimModifiers(modifierName) == trimModifiers(name) ||
trimModifiers(modifierName) == "by " + trimModifiers(name)
) {
if (makeActive) {
card.classList.add(activeCardClass)
card.querySelector(".modifier-card-image-overlay").innerText = "-"
} else {
card.classList.remove(activeCardClass)
card.querySelector(".modifier-card-image-overlay").innerText = "+"
}
}
2023-04-27 19:56:56 +02:00
})
}
function changePreviewImages(val) {
2023-04-27 19:56:56 +02:00
const previewImages = document.querySelectorAll(".modifier-card-image-container img")
let previewArr = []
2023-04-27 19:56:56 +02:00
modifiers.map((x) => x.modifiers).forEach((x) => previewArr.push(...x.map((m) => m.previews)))
previewArr = previewArr.map((x) => {
let obj = {}
2023-04-27 19:56:56 +02:00
x.forEach((preview) => {
obj[preview.name] = preview.path
})
2023-04-27 19:56:56 +02:00
return obj
})
2023-04-27 19:56:56 +02:00
previewImages.forEach((previewImage) => {
const currentPreviewType = previewImage.getAttribute("preview-type")
const relativePreviewPath = previewImage.src.split(modifierThumbnailPath + "/").pop()
2023-04-27 19:56:56 +02:00
const previews = previewArr.find((preview) => relativePreviewPath == preview[currentPreviewType])
2023-04-27 19:56:56 +02:00
if (typeof previews == "object") {
let preview = null
2023-04-27 19:56:56 +02:00
if (val == "portrait") {
preview = previews.portrait
2023-04-27 19:56:56 +02:00
} else if (val == "landscape") {
preview = previews.landscape
}
2023-04-27 19:56:56 +02:00
if (preview != null) {
previewImage.src = `${modifierThumbnailPath}/${preview}`
2023-04-27 19:56:56 +02:00
previewImage.setAttribute("preview-type", val)
}
}
})
}
function resizeModifierCards(val) {
2023-04-27 19:56:56 +02:00
const cardSizePrefix = "modifier-card-size_"
const modifierCardClass = "modifier-card"
const modifierCards = document.querySelectorAll(`.${modifierCardClass}`)
2023-04-27 19:56:56 +02:00
const cardSize = (n) => `${cardSizePrefix}${n}`
2023-04-27 19:56:56 +02:00
modifierCards.forEach((card) => {
// remove existing size classes
2023-04-27 19:56:56 +02:00
const classes = card.className.split(" ").filter((c) => !c.startsWith(cardSizePrefix))
card.className = classes.join(" ").trim()
2023-04-27 19:56:56 +02:00
if (val != 0) {
card.classList.add(cardSize(val))
}
})
}
modifierCardSizeSlider.onchange = () => resizeModifierCards(modifierCardSizeSlider.value)
previewImageField.onchange = () => changePreviewImages(previewImageField.value)
2022-10-21 10:38:16 +02:00
2023-04-27 19:56:56 +02:00
modifierSettingsBtn.addEventListener("click", function(e) {
2022-10-29 01:48:32 +02:00
modifierSettingsOverlay.classList.add("active")
customModifiersTextBox.setSelectionRange(0, 0)
customModifiersTextBox.focus()
customModifiersInitialContent = customModifiersTextBox.value // preserve the initial content
e.stopPropagation()
2022-10-21 10:38:16 +02:00
})
2023-04-27 19:56:56 +02:00
modifierSettingsOverlay.addEventListener("keydown", function(e) {
switch (e.key) {
case "Escape": // Escape to cancel
customModifiersTextBox.value = customModifiersInitialContent // undo the changes
modifierSettingsOverlay.classList.remove("active")
e.stopPropagation()
break
case "Enter":
2023-04-27 19:56:56 +02:00
if (e.ctrlKey) {
// Ctrl+Enter to confirm
modifierSettingsOverlay.classList.remove("active")
e.stopPropagation()
break
}
}
})
2022-10-21 10:38:16 +02:00
function saveCustomModifiers() {
localStorage.setItem(CUSTOM_MODIFIERS_KEY, customModifiersTextBox.value.trim())
loadCustomModifiers()
}
function loadCustomModifiers() {
2023-04-27 19:56:56 +02:00
PLUGINS["MODIFIERS_LOAD"].forEach((fn) => fn.loader.call())
2022-10-21 10:38:16 +02:00
}
2023-04-27 19:56:56 +02:00
customModifiersTextBox.addEventListener("change", saveCustomModifiers)