| | (function () { |
| |
|
| | if (!globalThis.LLuL) globalThis.LLuL = {}; |
| | const LLuL = globalThis.LLuL; |
| |
|
| | function id(type, s) { |
| | return `llul-${type}-${s}`; |
| | } |
| |
|
| | function isDark() { |
| | return gradioApp().querySelector('.dark') !== null; |
| | } |
| |
|
| | const M = 2; |
| | function setSize(canvas, width, height) { |
| | width = Math.floor(+width / M); |
| | height = Math.floor(+height / M); |
| | if (canvas.width != width) canvas.width = width; |
| | if (canvas.height != height) canvas.height = height; |
| | } |
| |
|
| | function updateXY(canvas) { |
| | let x = +canvas.dataset.x, |
| | y = +canvas.dataset.y, |
| | m = +canvas.dataset.m, |
| | mm = Math.pow(2, m), |
| | w = +canvas.width * M, |
| | h = +canvas.height * M; |
| | if (x < 0) x = 0; |
| | if (w < x + w / mm) x = Math.floor(w - w / mm); |
| | if (y < 0) y = 0; |
| | if (h < y + h / mm) y = Math.floor(h - h / mm); |
| |
|
| | canvas.dataset.x = x; |
| | canvas.dataset.y = y; |
| | canvas.dataset.m = m; |
| |
|
| | canvas.parentNode.querySelector('.llul-pos-x').value = x; |
| | canvas.parentNode.querySelector('.llul-pos-y').value = y; |
| | } |
| |
|
| | let last_image = new Image(); |
| | let hide_image = true; |
| | async function draw(canvas) { |
| | const |
| | x = +canvas.dataset.x, |
| | y = +canvas.dataset.y, |
| | m = +canvas.dataset.m, |
| | mm = Math.pow(2, m), |
| | w = +canvas.width, |
| | h = +canvas.height, |
| | bg = canvas.dataset.bg; |
| |
|
| | const ctx = canvas.getContext('2d'); |
| |
|
| | if (bg) { |
| | if (last_image?.src === bg) { |
| | |
| | } else { |
| | await (new Promise(resolve => { |
| | last_image.onload = () => resolve(); |
| | last_image.src = bg; |
| | })); |
| | } |
| | hide_image = false; |
| | } else { |
| | last_image.src = ''; |
| | hide_image = true; |
| | } |
| | |
| | if (last_image.src && !hide_image) { |
| | ctx.drawImage(last_image, 0, 0, +last_image.width, +last_image.height, 0, 0, +canvas.width, +canvas.height); |
| | } else { |
| | const bgcolor = isDark() ? 'black' : 'white'; |
| | ctx.fillStyle = bgcolor; |
| | ctx.fillRect(0, 0, +canvas.width, +canvas.height); |
| | } |
| |
|
| | |
| | const rectX = x / M; |
| | const rectY = y / M; |
| | const rectW = Math.floor(w / mm); |
| | const rectH = Math.floor(h / mm); |
| |
|
| | |
| | ctx.fillStyle = 'rgba(255, 255, 255, 0.2)'; |
| | ctx.fillRect(rectX, rectY, rectW, rectH); |
| |
|
| | |
| | ctx.strokeStyle = '#ff0000'; |
| | ctx.lineWidth = 2; |
| | ctx.strokeRect(rectX, rectY, rectW, rectH); |
| | |
| | |
| | ctx.beginPath(); |
| | ctx.moveTo(rectX + rectW / 2, rectY + rectH / 2 - 5); |
| | ctx.lineTo(rectX + rectW / 2, rectY + rectH / 2 + 5); |
| | ctx.moveTo(rectX + rectW / 2 - 5, rectY + rectH / 2); |
| | ctx.lineTo(rectX + rectW / 2 + 5, rectY + rectH / 2); |
| | ctx.stroke(); |
| | } |
| |
|
| | async function update_gradio(type, canvas) { |
| | await LLuL.js2py(type, 'x', +canvas.dataset.x); |
| | await LLuL.js2py(type, 'y', +canvas.dataset.y); |
| | } |
| |
|
| | function init(type) { |
| | const $$ = (x,n) => Array.from(gradioApp().querySelectorAll(x)).at(n); |
| | const $ = x => $$(x, -1); |
| | |
| | if (!$('#' + id(type, 'accordion'))) return false; |
| | |
| | const cont = $('#' + id(type, 'container')); |
| | const x = $('#' + id(type, 'x')); |
| | const y = $('#' + id(type, 'y')); |
| | const m = $(`#${id(type, 'm')} input[type=number]`); |
| | const ms = $(`#${id(type, 'm')} input[type=range]`); |
| | if (!cont || !x || !y || !m || !ms) return false; |
| |
|
| | if (cont.querySelector('canvas')) return true; |
| |
|
| | const width = $$(`#${type}_width input[type=number]`, 0); |
| | const height = $$(`#${type}_height input[type=number]`, 0); |
| | const width2 = $$(`#${type}_width input[type=range]`, 0); |
| | const height2 = $$(`#${type}_height input[type=range]`, 0); |
| |
|
| | const pos_x = Math.floor(+width.value / 4); |
| | const pos_y = Math.floor(+height.value / 4); |
| | |
| | const pos_cont = document.createElement('div'); |
| | pos_cont.innerHTML = ` |
| | <div class="llul-pos" id="llul-${type}-pos"> |
| | <label>x:<input type="number" value="${pos_x}" min="0" step="1" class="llul-pos-x" id="llul-${type}-pos-x"></label> |
| | <label>y:<input type="number" value="${pos_y}" min="0" step="1" class="llul-pos-y" id="llul-${type}-pos-y"></label> |
| | </div> |
| | `; |
| |
|
| | const canvas = document.createElement('canvas'); |
| | canvas.style.border = '1px solid gray'; |
| | canvas.dataset.x = pos_x; |
| | canvas.dataset.y = pos_y; |
| | canvas.dataset.m = m.value; |
| |
|
| | const bg_cont = document.createElement('div'); |
| | bg_cont.classList.add('llul-bg-setting'); |
| | bg_cont.innerHTML = ` |
| | <span>Load BG</span> |
| | <span>Erase BG</span> |
| | <input type="file" style="display:none"> |
| | `; |
| |
|
| | for (let ele of [width, height, width2, height2, m, ms]) { |
| | ele.addEventListener('input', e => { |
| | canvas.dataset.m = +m.value; |
| | setSize(canvas, width.value, height.value); |
| | updateXY(canvas); |
| | draw(canvas); |
| | }); |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| |
|
| | let dragging = false; |
| | let last_x, last_y; |
| | canvas.addEventListener('pointerdown', e => { |
| | e.preventDefault(); |
| | dragging = true; |
| | last_x = e.offsetX; |
| | last_y = e.offsetY; |
| | }); |
| | canvas.addEventListener('pointerup', async e => { |
| | e.preventDefault(); |
| | dragging = false; |
| | await update_gradio(type, canvas); |
| | }); |
| | canvas.addEventListener('pointermove', e => { |
| | if (!dragging) return; |
| | const dx = e.offsetX - last_x, dy = e.offsetY - last_y; |
| | const x = +canvas.dataset.x, y = +canvas.dataset.y; |
| | canvas.dataset.x = x + dx * M; |
| | canvas.dataset.y = y + dy * M; |
| | last_x = e.offsetX; |
| | last_y = e.offsetY; |
| | updateXY(canvas); |
| | draw(canvas); |
| | }); |
| | |
| | |
| | |
| | function set_bg(url) { |
| | canvas.dataset.bg = url; |
| | draw(canvas); |
| | } |
| | bg_cont.querySelector('input[type=file]').addEventListener('change', e => { |
| | const ele = e.target; |
| | const files = ele.files; |
| | if (files.length != 0) { |
| | const file = files[0]; |
| | const r = new FileReader(); |
| | r.onload = () => set_bg(r.result); |
| | r.readAsDataURL(file); |
| | } |
| | ele.value = ''; |
| | }, false); |
| | bg_cont.addEventListener('click', e => { |
| | const ele = e.target; |
| | if (ele.textContent == 'Load BG') { |
| | bg_cont.querySelector('input[type=file]').click(); |
| | } else if (ele.textContent == 'Erase BG') { |
| | set_bg(''); |
| | } |
| | }); |
| | |
| | |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | cont.appendChild(pos_cont); |
| | cont.appendChild(canvas); |
| | cont.appendChild(bg_cont); |
| | setSize(canvas, width.value, height.value); |
| | updateXY(canvas); |
| | draw(canvas); |
| |
|
| | return true; |
| | } |
| |
|
| | function init2(type, init_fn) { |
| | const repeat_until = (fn, resolve) => { |
| | const v = fn(); |
| | if (v) { |
| | resolve(v); |
| | } else { |
| | setTimeout(() => repeat_until(fn, resolve), 500); |
| | } |
| | }; |
| | |
| | return new Promise(resolve => repeat_until(() => init_fn(type), resolve)); |
| | } |
| | |
| | function init_LLuL() { |
| | if (!LLuL.txt2img) { |
| | LLuL.txt2img = init2('txt2img', init); |
| | if (LLuL.txt2img) { |
| | LLuL.txt2img.then(() => console.log('[LLuL] txt2img initialized')); |
| | } |
| | } |
| |
|
| | if (!LLuL.img2img) { |
| | LLuL.img2img = init2('img2img', init); |
| | if (LLuL.img2img) { |
| | LLuL.img2img.then(() => console.log('[LLuL] img2img initialized')); |
| | } |
| | } |
| |
|
| | return LLuL.txt2img && LLuL.img2img; |
| | } |
| |
|
| | function apply() { |
| | const ok = init_LLuL(); |
| | if (!ok) { |
| | setTimeout(apply, 500); |
| | } |
| | } |
| |
|
| | apply(); |
| |
|
| | })(); |
| |
|