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

521 lines
22 KiB
HTML

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>oven</title>
<link rel="icon" href="favicon.png">
<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>
<body onload="init()">
<div id="left" ondragover="event.preventDefault()">
<h1>of <span class="em">in</span> the cold food</h1>
<div class="imagewrap">
<img id="imgin" src=""></img>
</div>
<input type="file" id="input" name="img" accept="image/*" />
<p>alternatively, <span class="em">paste</span> an image from the clipboard*</p>
<p style="font-size:.8em; font-style: italic; color: #888; margin-top: -10px;">* doesn't seem to work with images copied from the browser. known issue, makes my brain hurt</p>
</div>
<div id="convert">
<button id="btn" onclick="bake()">bake</button>
<canvas id="preview" width="208" height="16"></canvas>
<span class="em">palette</span><br/>
<input type="radio" id="gg" name="palette" onchange="updatePreview()" checked /><label for="gg"><img src="" style="vertical-align: middle; transform: scale(-1)"/> grey and grey</label><br/>
<input type="radio" id="bw" name="palette" onchange="updatePreview()" /><label for="bw"><img src="" style="vertical-align: middle; transform: scale(-1)"/> black and white</label><br/>
<br/>
<span class="em">dither pattern</span><br/>
<input type="radio" id="nd" name="ditherpat" onchange="checkdpt()" /><label for="nd">no dithering</label><br/>
<input type="radio" id="p3" name="ditherpat" onchange="checkdpt()" /><label for="p3">3 patterns</label><br/>
<input type="radio" id="p5" name="ditherpat" onchange="checkdpt()" /><label for="p5">5 patterns</label><br/>
<input type="radio" id="p9" name="ditherpat" onchange="checkdpt()" /><label for="p9">9 patterns</label><br/>
<input type="radio" id="p17"name="ditherpat" onchange="checkdpt()" checked /><label for="p17">17 patterns</label><br/>
<input type="radio" id="bn" name="ditherpat" onchange="checkdpt()" /><label for="bn">blue noise</label><br/>
<br/>
<span id="dpt"><span class="em">dither pattern type</span><br/>
<input type="radio" id="cheq" name="dithertype" onchange="updatePreview()" checked /><label for="cheq">chequered</label><br/>
<input type="radio" id="hatc" name="dithertype" onchange="updatePreview()" /><label for="hatc">hatched</label><br/>
<span id="hybrid"><input type="radio" id="hybr" onchange="updatePreview()" name="dithertype" /><label for="hybr">hybrid</label></span><br/>
</span>
<br/>
<span class="em">dither block size</span><br/>
<input type="radio" id="d11" name="dithersize" onchange="updatePreview()" checked /><label for="d11">1&times;1 pixels</label><br/>
<input type="radio" id="d22" name="dithersize" onchange="updatePreview()" /><label for="d22">2&times;2 pixels</label><br/>
<input type="radio" id="d44" name="dithersize" onchange="updatePreview()" /><label for="d44">4&times;4 pixels</label><br/><br/>
<span class="em">what to use for greyscale</span><br/>
<input type="radio" id="greylight" name="greytype" onchange="updatePreview()" /><label for="greylight">lightness</label><br/>
<input type="radio" id="greyvalue" name="greytype" onchange="updatePreview()" /><label for="greyvalue">value</label><br/>
<input type="radio" id="greylstar" name="greytype" onchange="updatePreview()" checked /><label for="greylstar">L* (D65/2°)</label><br/>
<br/>
<span class="em">curves</span><br/>
<div style="display:block;height:218px">
<canvas id="curvebg" width="208" height="208" style="position:absolute;margin-top:3px"></canvas>
<canvas id="curves" width="208" height="208" style="position:absolute;margin-top:3px"></canvas>
</div>
x to the power <input type="number" id="exp" min="0.1" max="5" step="0.1" style="width:40px;color:#222" onchange="setPower()" value="1"></input>
<button onclick="exp.value = 1; exp.onchange()">reset</button>
</div>
<div id="right">
<h1>of <span class="em">out</span> hot eat the food</h1>
<div class="imagewrap">
<canvas id="canv" width="1" height="1"></canvas>
</div>
<p style="font-size:.9em; font-style: italic; color: #888;">(right click the output to copy or save it, save/copy buttons are in the works)</p>
</div>
</body>
</html>