521 lines
22 KiB
HTML
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×1 pixels</label><br/>
|
|
<input type="radio" id="d22" name="dithersize" onchange="updatePreview()" /><label for="d22">2×2 pixels</label><br/>
|
|
<input type="radio" id="d44" name="dithersize" onchange="updatePreview()" /><label for="d44">4×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>
|