forked from extern/easydiffusion
318 lines
18 KiB
JavaScript
318 lines
18 KiB
JavaScript
;(function(){
|
||
"use strict";
|
||
const PAPERSIZE = [
|
||
{id: "a3p", width: 297, height: 420, unit: "mm"},
|
||
{id: "a3l", width: 420, height: 297, unit: "mm"},
|
||
{id: "a4p", width: 210, height: 297, unit: "mm"},
|
||
{id: "a4l", width: 297, height: 210, unit: "mm"},
|
||
{id: "ll", width: 279, height: 216, unit: "mm"},
|
||
{id: "lp", width: 216, height: 279, unit: "mm"},
|
||
{id: "hd", width: 1920, height: 1080, unit: "pixels"},
|
||
{id: "4k", width: 3840, height: 2160, unit: "pixels"},
|
||
]
|
||
|
||
// ---- Register plugin
|
||
PLUGINS['IMAGE_INFO_BUTTONS'].push({
|
||
html: '<i class="fa-solid fa-table-cells-large"></i> Download tiled image',
|
||
on_click: onDownloadTiledImage,
|
||
filter: (req, img) => req.tiling != "none",
|
||
})
|
||
|
||
var thisImage
|
||
|
||
function onDownloadTiledImage(req, img) {
|
||
document.getElementById("download-tiled-image-dialog").showModal()
|
||
thisImage = new Image()
|
||
thisImage.src = img.src
|
||
thisImage.dataset["prompt"] = img.dataset["prompt"]
|
||
}
|
||
|
||
// ---- Add HTML
|
||
document.getElementById('container').lastElementChild.insertAdjacentHTML("afterend",
|
||
`<dialog id="download-tiled-image-dialog">
|
||
<div class="dialog-header">
|
||
<div class="dialog-header-left">
|
||
<h4>Download tiled image</h4>
|
||
<span>Generate a larger image from this tile</span>
|
||
</div>
|
||
<div id="download-header-right">
|
||
<i id="downnload-tiled-close-button" class="fa-solid fa-xmark fa-lg"></i>
|
||
</div>
|
||
</div>
|
||
<div class="download-tiled-image dtim-container">
|
||
<div class="download-tiled-image-top">
|
||
<div class="tab-container">
|
||
<span id="tab-image-tiles" class="tab active">
|
||
<span>Number of tiles</small></span>
|
||
</span>
|
||
<span id="tab-image-size" class="tab">
|
||
<span>Image dimensions</span>
|
||
</span>
|
||
</div>
|
||
<div>
|
||
<div id="tab-content-image-tiles" class="tab-content active">
|
||
<div class="tab-content-inner">
|
||
<label for="dtim1-width">Width:</label> <input id="dtim1-width" min="1" max="99" type="number" value="2">
|
||
<label for="dtim1-height">Height:</label> <input id="dtim1-height" min="1" max="99" type="number" value="2">
|
||
</div>
|
||
</div>
|
||
<div id="tab-content-image-size" class="tab-content">
|
||
<div class="tab-content-inner">
|
||
<div class="method-2-options">
|
||
<label for="dtim2-width">Width:</label> <input id="dtim2-width" size="3" value="1920">
|
||
<label for="dtim2-height">Height:</label> <input id="dtim2-height" size="3" value="1080">
|
||
<select id="dtim2-unit">
|
||
<option>pixels</option>
|
||
<option>mm</option>
|
||
<option>inches</option>
|
||
</select>
|
||
</div>
|
||
<div class="method-2-dpi">
|
||
<label for="dtim2-dpi">DPI:</label> <input id="dtim2-dpi" size="3" value="72">
|
||
</div>
|
||
<div class="method-2-paper">
|
||
<i>Some standard sizes:</i><br>
|
||
<button id="dtim2-a3p">A3 portrait</button><button id="dtim2-a3l">A3 landscape</button><br>
|
||
<button id="dtim2-a4p">A4 portrait</button><button id="dtim2-a4l">A4 landscape</button><br>
|
||
<button id="dtim2-lp">Letter portrait</button><button id="dtim2-ll">Letter landscape</button><br>
|
||
<button id="dtim2-hd">Full HD</button><button id="dtim2-4k">4K</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="download-tiled-image-placement">
|
||
<div class="tab-container">
|
||
<span id="tab-image-placement" class="tab active">
|
||
<span>Tile placement</span>
|
||
</span>
|
||
</div>
|
||
<div>
|
||
<div id="tab-content-image-placement" class="tab-content active">
|
||
<div class="tab-content-inner">
|
||
<img id="dtim-1tl" class="active" src="" />
|
||
<img id="dtim-1tr" src="" /><br>
|
||
<img id="dtim-1bl" src="" />
|
||
<img id="dtim-1br" src="" /> <br>
|
||
<img id="dtim-1center" src="" />
|
||
<img id="dtim-4center" src="" /> <br>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="dtim-ok">
|
||
<button class="primaryButton" id="dti-ok">Download</button>
|
||
</div>
|
||
<div class="dtim-newtab">
|
||
<button class="primaryButton" id="dti-newtab">Open in new tab</button>
|
||
</div>
|
||
<div class="dtim-cancel">
|
||
<button class="primaryButton" id="dti-cancel">Cancel</button>
|
||
</div>
|
||
</div>
|
||
</dialog>`)
|
||
|
||
let downloadTiledImageDialog = document.getElementById("download-tiled-image-dialog")
|
||
let dtim1_width = document.getElementById("dtim1-width")
|
||
let dtim1_height = document.getElementById("dtim1-height")
|
||
let dtim2_width = document.getElementById("dtim2-width")
|
||
let dtim2_height = document.getElementById("dtim2-height")
|
||
let dtim2_unit = document.getElementById("dtim2-unit")
|
||
let dtim2_dpi = document.getElementById("dtim2-dpi")
|
||
let tabTiledTilesOptions = document.getElementById("tab-image-tiles")
|
||
let tabTiledSizeOptions = document.getElementById("tab-image-size")
|
||
|
||
linkTabContents(tabTiledTilesOptions)
|
||
linkTabContents(tabTiledSizeOptions)
|
||
|
||
prettifyInputs(downloadTiledImageDialog)
|
||
|
||
// ---- Predefined image dimensions
|
||
PAPERSIZE.forEach( function(p) {
|
||
document.getElementById("dtim2-" + p.id).addEventListener("click", (e) => {
|
||
dtim2_unit.value = p.unit
|
||
dtim2_width.value = p.width
|
||
dtim2_height.value = p.height
|
||
})
|
||
})
|
||
|
||
// ---- Close popup
|
||
document.getElementById("dti-cancel").addEventListener("click", (e) => downloadTiledImageDialog.close())
|
||
document.getElementById("downnload-tiled-close-button").addEventListener("click", (e) => downloadTiledImageDialog.close())
|
||
modalDialogCloseOnBackdropClick(downloadTiledImageDialog)
|
||
makeDialogDraggable(downloadTiledImageDialog)
|
||
|
||
// ---- Stylesheet
|
||
const styleSheet = document.createElement("style")
|
||
styleSheet.textContent = `
|
||
button[disabled] {
|
||
opacity: 0.5;
|
||
}
|
||
|
||
.method-2-dpi {
|
||
margin-top: 1em;
|
||
margin-bottom: 1em;
|
||
}
|
||
|
||
.method-2-paper button {
|
||
width: 10em;
|
||
padding: 4px;
|
||
margin: 4px;
|
||
}
|
||
|
||
.download-tiled-image .tab-content {
|
||
background: var(--background-color1);
|
||
border-radius: 3pt;
|
||
}
|
||
|
||
.dtim-container { display: grid;
|
||
grid-template-columns: auto auto;
|
||
grid-template-rows: auto auto;
|
||
gap: 1em 0px;
|
||
grid-auto-flow: row;
|
||
grid-template-areas:
|
||
"dtim-tab dtim-tab dtim-plc"
|
||
"dtim-ok dtim-newtab dtim-cancel";
|
||
}
|
||
|
||
.download-tiled-image-top {
|
||
justify-self: center;
|
||
grid-area: dtim-tab;
|
||
}
|
||
|
||
.download-tiled-image-placement {
|
||
justify-self: center;
|
||
grid-area: dtim-plc;
|
||
margin-left: 1em;
|
||
}
|
||
|
||
.dtim-ok {
|
||
justify-self: center;
|
||
align-self: start;
|
||
grid-area: dtim-ok;
|
||
}
|
||
|
||
.dtim-newtab {
|
||
justify-self: center;
|
||
align-self: start;
|
||
grid-area: dtim-newtab;
|
||
}
|
||
|
||
.dtim-cancel {
|
||
justify-self: center;
|
||
align-self: start;
|
||
grid-area: dtim-cancel;
|
||
}
|
||
|
||
#tab-content-image-placement img {
|
||
margin: 4px;
|
||
opacity: 0.3;
|
||
border: solid 2px var(--background-color1);
|
||
}
|
||
|
||
#tab-content-image-placement img:hover {
|
||
margin: 4px;
|
||
opacity: 1;
|
||
border: solid 2px var(--accent-color);
|
||
filter: brightness(2);
|
||
}
|
||
|
||
#tab-content-image-placement img.active {
|
||
margin: 4px;
|
||
opacity: 1;
|
||
border: solid 2px var(--background-color1);
|
||
}
|
||
|
||
`
|
||
document.head.appendChild(styleSheet)
|
||
|
||
// ---- Placement widget
|
||
|
||
function updatePlacementWidget(event) {
|
||
document.querySelector("#tab-content-image-placement img.active").classList.remove("active")
|
||
event.target.classList.add("active")
|
||
}
|
||
|
||
document.querySelectorAll("#tab-content-image-placement img").forEach(
|
||
(i) => i.addEventListener("click", updatePlacementWidget)
|
||
)
|
||
|
||
function getPlacement() {
|
||
return document.querySelector("#tab-content-image-placement img.active").id.substr(5)
|
||
}
|
||
|
||
// ---- Make the image
|
||
function downloadTiledImage(image, width, height, offsetX=0, offsetY=0, new_tab=false) {
|
||
|
||
const canvas = document.createElement('canvas')
|
||
canvas.width = width
|
||
canvas.height = height
|
||
const context = canvas.getContext('2d')
|
||
|
||
const w = image.width
|
||
const h = image.height
|
||
|
||
for (var x = offsetX; x < width; x += w) {
|
||
for (var y = offsetY; y < height; y += h) {
|
||
context.drawImage(image, x, y, w, h)
|
||
}
|
||
}
|
||
if (new_tab) {
|
||
var newTab = window.open("")
|
||
newTab.document.write(`<html><head><title>${width}×${height}, "${image.dataset["prompt"]}"</title></head><body><img src="${canvas.toDataURL()}"></body></html>`)
|
||
} else {
|
||
const link = document.createElement('a')
|
||
link.href = canvas.toDataURL()
|
||
link.download = image.dataset["prompt"].replace(/[^a-zA-Z0-9]+/g, "-").substr(0,22)+crypto.randomUUID()+".png"
|
||
link.click()
|
||
}
|
||
}
|
||
|
||
function onDownloadTiledImageClick(e, newtab=false) {
|
||
var width, height, offsetX, offsetY
|
||
|
||
if (isTabActive(tabTiledTilesOptions)) {
|
||
width = thisImage.width * dtim1_width.value
|
||
height = thisImage.height * dtim1_height.value
|
||
} else {
|
||
if ( dtim2_unit.value == "pixels" ) {
|
||
width = dtim2_width.value
|
||
height= dtim2_height.value
|
||
} else if ( dtim2_unit.value == "mm" ) {
|
||
width = Math.floor( dtim2_width.value * dtim2_dpi.value / 25.4 )
|
||
height = Math.floor( dtim2_height.value * dtim2_dpi.value / 25.4 )
|
||
} else { // inch
|
||
width = Math.floor( dtim2_width.value * dtim2_dpi.value )
|
||
height = Math.floor( dtim2_height.value * dtim2_dpi.value )
|
||
}
|
||
}
|
||
|
||
var placement = getPlacement()
|
||
if (placement == "1tl") {
|
||
offsetX = 0
|
||
offsetY = 0
|
||
} else if (placement == "1tr") {
|
||
offsetX = width - thisImage.width * Math.ceil( width / thisImage.width )
|
||
offsetY = 0
|
||
} else if (placement == "1bl") {
|
||
offsetX = 0
|
||
offsetY = height - thisImage.height * Math.ceil( height / thisImage.height )
|
||
} else if (placement == "1br") {
|
||
offsetX = width - thisImage.width * Math.ceil( width / thisImage.width )
|
||
offsetY = height - thisImage.height * Math.ceil( height / thisImage.height )
|
||
} else if (placement == "4center") {
|
||
offsetX = width/2 - thisImage.width * Math.ceil( width/2 / thisImage.width )
|
||
offsetY = height/2 - thisImage.height * Math.ceil( height/2 / thisImage.height )
|
||
} else if (placement == "1center") {
|
||
offsetX = width/2 - thisImage.width/2 - thisImage.width * Math.ceil( (width/2 - thisImage.width/2) / thisImage.width )
|
||
offsetY = height/2 - thisImage.height/2 - thisImage.height * Math.ceil( (height/2 - thisImage.height/2) / thisImage.height )
|
||
}
|
||
downloadTiledImage(thisImage, width, height, offsetX, offsetY, newtab)
|
||
downloadTiledImageDialog.close()
|
||
}
|
||
|
||
document.getElementById("dti-ok").addEventListener("click", onDownloadTiledImageClick)
|
||
document.getElementById("dti-newtab").addEventListener("click", (e) => onDownloadTiledImageClick(e,true))
|
||
|
||
})()
|