Files
oven/index.html
2024-06-07 13:19:02 +03:00

22 KiB
Raw Permalink Blame History

<html> <head> <style> body { background-color: #222; margin: 10px; font-family: sans-serif; color: #ccc; } #left, #right { display: block; position: fixed; top: 10px; box-sizing: border-box; border-radius: 20px; border: 2px solid #555; width: calc(50vw - 144px); height: calc(100vh - 20px); padding: 0 20px 0 20px; overflow-y: auto; } #left { left: 10px; } #right { right: 10px; } #convert { display: block; box-sizing: border-box; margin-left: 50%; transform: translate(-50%, 0); width: 250px; padding: 20px 20px 0 20px; } h1, p, input, label { color: #ccc; font-family: sans-serif; } .imagewrap { position: relative; width: 100%; border: 2px solid #333; overflow: auto; margin-bottom: 20px; background-color: #1c1c1c; min-height: 240px; } .em { font-weight: bold; color: #ee0; } #btn { width: 100%; height: 40px; font-size: 1.7em; font-family: sans-serif; margin-bottom: 20px; } label img, #convert canvas { border:1px solid #666; } label { margin-left: 2px; } .disabled { opacity: 0.5; pointer-events: none; } .disabled .disabled { opacity: 1; } </style> <script> let imageIn = document.createElement("img"); let ctx, ct2, cc; let img64 = document.createElement("img"); let dragcurve = false; let mtx17c = [[16/17, 8/17, 14/17, 6/17], [ 4/17, 12/17, 2/17, 10/17], [13/17, 5/17, 15/17, 7/17], [ 1/17, 9/17, 3/17, 11/17]]; let mtx17h = [[16/17, 12/17, 15/17, 10/17], [ 4/17, 8/17, 3/17, 6/17], [13/17, 11/17, 14/17, 9/17], [ 1/17, 7/17, 2/17, 5/17]]; let mtx17x = [[16/17, 12/17, 14/17, 10/17], [ 4/17, 8/17, 2/17, 6/17], [13/17, 9/17, 15/17, 11/17], [ 1/17, 5/17, 3/17, 7/17]]; let mtx9c = [[8/9, 4/9, 7/9, 3/9], [2/9, 6/9, 1/9, 5/9], [7/9, 3/9, 8/9, 4/9], [1/9, 5/9, 2/9, 6/9]]; let mtx9h = [[8/9, 6/9, 8/9, 5/9], [2/9, 4/9, 2/9, 3/9], [7/9, 6/9, 7/9, 5/9], [1/9, 4/9, 1/9, 3/9]]; let mtx9x = [[8/9, 5/9, 7/9, 6/9], [2/9, 3/9, 1/9, 4/9], [7/9, 6/9, 8/9, 5/9], [1/9, 4/9, 2/9, 3/9]]; let mtx5c = [[4/5, 2/5], [1/5, 3/5]]; let mtx5h = [[4/5, 3/5], [1/5, 2/5]]; let mtx3c = [[2/3, 1/3], [1/3, 2/3]]; let mtx3h = [[2/3, 2/3], [1/3, 1/3]]; let mtx1 = 1/2; let mtxblue = []; let i256 = []; let curve = []; img64.onload = function() { let c = document.createElement("canvas"); c.width = 64; c.height = 64; let ct = c.getContext("2d"); ct.drawImage(img64, 0, 0); let data = ct.getImageData(0, 0, 64, 64).data; i = 0; for (let y = 0; y < 64; y++) { let arr = []; for (let x = 0; x < 64; x++, i+=4) { arr.push(data[i]/255); } mtxblue.push(arr); } }; img64.src = ""; function init() { input.addEventListener("change", handleFiles, false); ctx = canv.getContext("2d"); ct2 = preview.getContext("2d"); cc = curves.getContext("2d"); for (let i = 0; i < 256; i++) i256.push(i), curve.push(i); checkdpt(); updateCurve(); let c = curvebg.getContext("2d"); c.fillStyle = "#333"; c.fillRect(curvebg.width/4,0,1,curvebg.height); c.fillRect(curvebg.width/4*3,0,1,curvebg.height); c.fillRect(0,curvebg.height/4,curvebg.width,1); c.fillRect(0,curvebg.height/4*3,curvebg.width,1); c.fillStyle = "#444"; c.fillRect(curvebg.width/2,0,1,curvebg.height); c.fillRect(0,curvebg.height/2,curvebg.width,1); c.fillStyle = "#666"; c.font = '16px sans-serif'; c.textAlign = "right"; c.fillText("input", curvebg.width - 4, curvebg.height - 8); c.textAlign = "left"; c.fillText("output", 5, 16); let grad = cc.createLinearGradient(0, curvebg.height, curvebg.width/2, curvebg.height/2); grad.addColorStop(0, 'black'); grad.addColorStop(1, 'white'); c.fillStyle = grad; c.fillRect(0,0,1,curvebg.height); c.fillRect(0,curvebg.height - 1,curvebg.width,1); curves.onmousedown = (e) => { dragcurve = true; let b = curves.getBoundingClientRect(); let x = (e.x - b.x) < 0 ? 0 : (e.x - b.x) > curves.width ? curves.width : (e.x - b.x); let y = (e.y - b.y) < 0 ? 0 : (e.y - b.y) > curves.height ? curves.height : (e.y - b.y); x = Math.round(x / curves.width * 63); y = 63 - Math.round(y / curves.height * 63); for (let i = 0; i < 4; i++) curve[x * 4 + i] = y * 4; updateCurve(); }; window.onmousemove = (e) => { if (!dragcurve) return; let b = curves.getBoundingClientRect(); let x = (e.x - b.x) < 0 ? 0 : (e.x - b.x) > curves.width ? curves.width : (e.x - b.x); let y = (e.y - b.y) < 0 ? 0 : (e.y - b.y) > curves.height ? curves.height : (e.y - b.y); x = Math.round(x / curves.width * 63); y = 63 - Math.round(y / curves.height * 63); for (let i = 0; i < 4; i++) curve[x * 4 + i] = y * 4; updateCurve(); }; window.onmouseup = (e) => { dragcurve = false; }; window.onmouseleave = (e) => { dragcurve = false; }; exp.value = 1; } function handleFiles() { const fileList = this.files; console.log(this.files); imgin.src = URL.createObjectURL(this.files[0]); imageIn.src = URL.createObjectURL(this.files[0]); } window.addEventListener("paste", async function(e) { e.preventDefault(); e.stopPropagation(); e.clipboardData.items[0].getAsString(s => console.log(s)); let file = e.clipboardData.items[0].getAsFile(); imageIn.onload = function() { imgin.src = imageIn.src; URL.revokeObjectURL(imageIn.src); }; if ('srcObject' in imageIn) { imageIn.srcObject = file; } else { imageIn.src = URL.createObjectURL(file); } }); function bake() { canv.width = imageIn.width; canv.height = imageIn.height; ctx.imageSmoothingEnabled = false; let scale = d11.checked ? 1 : d22.checked ? 2 : d44.checked ? 4 : 1; let mtx = p17.checked && cheq.checked ? mtx17c : p17.checked && hatc.checked ? mtx17h : p17.checked && hybr.checked ? mtx17x : p9.checked && cheq.checked ? mtx9c : p9.checked && hatc.checked ? mtx9h : p9.checked && hybr.checked ? mtx9x : p5.checked && cheq.checked ? mtx5c : p5.checked && hatc.checked ? mtx5h : p3.checked && cheq.checked ? mtx3c : p3.checked && hatc.checked ? mtx3h : bn.checked ? mtxblue : mtx1; ctx.drawImage(imgin, 0, 0, canv.width / scale, canv.height / scale); ctx.drawImage(canv, 0, 0, canv.width / scale, canv.height / scale, 0, 0, canv.width, canv.height); data = ctx.getImageData(0, 0, canv.width, canv.height); let w = mtx[0].length, h = mtx.length; let white = gg.checked ? [0xb1, 0xae, 0xa8] : [0xff, 0xff, 0xff]; let black = gg.checked ? [0x32, 0x2f, 0x29] : [0, 0, 0]; let func = greylight.checked ? getLightness : greyvalue.checked ? getValue : greylstar.checked ? getLstar : getLightness; let i = 0, grey; for (let y = 0; y < canv.height; y++) { for (let x = 0; x < canv.width; x++, i += 4) { grey = func(data.data[i], data.data[i+1], data.data[i+2]); grey = Math.round(grey * 255); if (grey < 0) grey = 0; else if (grey > 255) grey = 255; grey = curve[grey]; grey /= 255; if (grey > mtx[y%h][x%w]) { // white data.data[i] = white[0]; data.data[i+1] = white[1]; data.data[i+2] = white[2]; } else { // black data.data[i] = black[0]; data.data[i+1] = black[1]; data.data[i+2] = black[2]; } } } ctx.putImageData(data, 0, 0); } function getLstar(r, g, b) { // D65/2° let Y, yr; r /= 255; g /= 255; b /= 255; r = ((r > 0.04045) ? Math.pow((r+0.055)/1.055,2.4) : r/12.92) * 100; g = ((g > 0.04045) ? Math.pow((g+0.055)/1.055,2.4) : g/12.92) * 100; b = ((b > 0.04045) ? Math.pow((b+0.055)/1.055,2.4) : b/12.92) * 100; Y = (0.2126*r + 0.7152*g + 0.0722*b) / 100; yr = (Y > 0.008856) ? Math.pow(Y, 1/3) : (7.787 * Y) + 16/116; let L = (116*yr)-16; return L/100; } function getLightness(r, g, b) { r /= 255; g /= 255; b /= 255; let Cmax = Math.max(r, g, b); let Cmin = Math.min(r, g, b); return (Cmax + Cmin) / 2; } function getValue(r, g, b) { return Math.max(r, g, b) / 255; } function checkdpt() { cheq.disabled = false; hatc.disabled = false; hybr.disabled = false; if (!(p3.checked || p5.checked || p9.checked || p17.checked)) { cheq.disabled = true; hatc.disabled = true; hybr.disabled = true; } if (p3.checked || p5.checked) { if (hybr.checked) hatc.checked = true; hybr.disabled = true; } dpt.classList.toggle("disabled", !(p3.checked || p5.checked || p9.checked || p17.checked)); hybrid.classList.toggle("disabled", !(p9.checked || p17.checked)); updatePreview(); } function updatePreview() { data = ct2.createImageData(preview.width, preview.height); ctx.imageSmoothingEnabled = false; let scale = d11.checked ? 1 : d22.checked ? 2 : d44.checked ? 4 : 1; let mtx = p17.checked && cheq.checked ? mtx17c : p17.checked && hatc.checked ? mtx17h : p17.checked && hybr.checked ? mtx17x : p9.checked && cheq.checked ? mtx9c : p9.checked && hatc.checked ? mtx9h : p9.checked && hybr.checked ? mtx9x : p5.checked && cheq.checked ? mtx5c : p5.checked && hatc.checked ? mtx5h : p3.checked && cheq.checked ? mtx3c : p3.checked && hatc.checked ? mtx3h : bn.checked ? mtxblue : mtx1; let grad = ct2.createLinearGradient(0, 0, preview.width / scale, 0); grad.addColorStop(0, 'black'); grad.addColorStop(1, 'white'); ct2.fillStyle = grad; ct2.fillRect(0, 0, preview.width / scale, preview.height / scale); ct2.drawImage(preview, 0, 0, preview.width / scale, preview.height / scale, 0, 0, preview.width, preview.height); data = ct2.getImageData(0, 0, preview.width, preview.height); let w = mtx[0].length, h = mtx.length; let white = gg.checked ? [0xb1, 0xae, 0xa8] : [0xff, 0xff, 0xff]; let black = gg.checked ? [0x32, 0x2f, 0x29] : [0, 0, 0]; let func = greylight.checked ? getLightness : greyvalue.checked ? getValue : greylstar.checked ? getLstar : getLightness; let i = 0, grey; for (let y = 0; y < preview.height; y++) { for (let x = 0; x < preview.width; x++, i += 4) { grey = func(data.data[i], data.data[i+1], data.data[i+2]); grey = Math.round(grey * 255); if (grey < 0) grey = 0; else if (grey > 255) grey = 255; grey = curve[grey]; grey /= 255; if (grey > mtx[y%h][x%w]) { // white data.data[i] = white[0]; data.data[i+1] = white[1]; data.data[i+2] = white[2]; } else { // black data.data[i] = black[0]; data.data[i+1] = black[1]; data.data[i+2] = black[2]; } } } ct2.putImageData(data, 0, 0); grad = ct2.createLinearGradient(0, 0, preview.width, 0); grad.addColorStop(0, 'black'); grad.addColorStop(1, 'white'); ct2.fillStyle = grad; ct2.fillRect(0, 0, preview.width, preview.height >> 1); } function updateCurve() { cc.clearRect(0,0,curves.width,curves.height); cc.strokeStyle = "#ee0"; cc.lineWidth = 2; cc.beginPath(); cc.moveTo(0, curves.height); for (let i = 0; i < 255; i++) { cc.lineTo(i/255 * curves.width + 1, (1 - curve[i]/255) * curves.height - 1); } cc.stroke(); cc.clearRect(0,0,1,curves.height); cc.clearRect(0,curves.height - 1,curves.width,1); updatePreview(); } function setCurve(c) { for (let i = 0; i < 256; i++) { curve[i] = c[i]; } updateCurve(); } function setPower() { let p = exp.value; for (let i = 0; i < 256; i++) { curve[i] = Math.round(Math.pow(i/255, p) * 255); } updateCurve(); } </script> </head>

of in the cold food

alternatively, paste an image from the clipboard*

* doesn't seem to work with images copied from the browser. known issue, makes my brain hurt

bake palette



dither pattern







dither pattern type




dither block size




what to use for greyscale




curves
x to the power reset

of out hot eat the food

(right click the output to copy or save it, save/copy buttons are in the works)

</html>