Name | Author | Game Mode | Rating | |||||
---|---|---|---|---|---|---|---|---|
True Fur | Violet CLM | Mutator | 9.5 |
<!doctype html>
<html>
<head>
<meta charset="windows-1252">
<title>Fur color designer for TrueFur.mut</title>
<style>
main {
max-width: 640px;
margin: 0 auto;
}
#preview {
margin-top: 0.5em;
padding-top: calc(100% * 103/120);
background-size: cover;
image-rendering: pixelated;
}
button {
width: 20%;
display: inline-block;
}
ul {
list-style: none;
padding-left: 0;
}
#gradients > li {
width: 50%;
display: inline-block;
}
ul.gradient {
width: 100%;
}
ul.gradient > li {
display: inline-block;
width: 11%;
height: 0;
padding-top: calc(11% - 6px);
position: relative;
box-sizing: border-box;
}
ul.gradient > li + li {
border: 3px inset;
position: relative;
}
li.enabled {
border-style: outset !important;
}
ul.gradient > li + li:not(.enabled):after {
content: url("");
position: absolute;
bottom: -3px;
right: -3px;
mix-blend-mode: luminosity;
}
li.enabled input[type=color] {
opacity: 0;
visibility: visible;
cursor: pointer;
}
select {
position: absolute;
top: -3px; bottom: 3px;
width: 100%;
}
input[type=color] {
position: absolute;
width: 100%;
top: 0; left: 0; bottom: 0; right: 0;
box-sizing: border-box;
visibility: hidden;
}
input[type=file] {
display: none;
}
#buttons {
position: sticky;
top: 0;
}
</style>
</head>
<body
><main
><div id="buttons"
><input type="file" id="hiddenload" accept=".asdat"
><button id="butload">Load</button
><button class="butsave">Save Player 1</button
><button class="butsave">Save Player 2</button
><button class="butsave">Save Player 3</button
><button class="butsave">Save Player 4</button
></div
><div id="preview"></div
><ul id="gradients"
></ul
></main>
<script>
const FILEVERSION = 0;
window.addEventListener('load', function() {
const CanonGradients = {
Green: [199,255,0, 147,223,0, 107,191,0, 71,163,0, 43,131,0, 19,103,0, 7,55,0, 0,11,0],
Red: [255,0,0, 227,0,0, 199,0,0, 171,0,0, 143,0,0, 115,0,0, 63,0,0, 11,0,0],
Blue: [187,227,255, 123,199,255, 59,171,255, 0,139,255, 0,107,203, 0,79,151, 0,47,79, 0,7,11],
Orange: [255,255,0, 255,199,0, 255,147,0, 255,95,0, 203,55,0, 155,27,0, 83,7,0, 11,0,0],
Pink: [251,139,183, 247,91,151, 243,43,123, 239,0,99, 191,0,75, 147,0,55, 99,0,35, 55,0,19],
Yellow: [255,255,0, 240,235,0, 230,210,0, 219,195,0, 187,147,0, 155,107,0, 83,55,0, 11,7,0],
Brown: [255,243,211, 219,207,175, 187,175,147, 155,139,115, 119,107,87, 87,75,63, 47,35,31, 11,7,7],
Silver: [211,231,255, 171,195,219, 139,159,187, 107,127,155, 75,95,119, 51,63,87, 27,31,47, 7,7,11],
Greenblue: [0,255,163, 0,227,127, 7,199,95, 7,171,67, 11,143,47, 11,119,31, 0,63,7, 0,11,0],
Purple: [231,119,255, 231,71,239, 223,31,207, 207,0,163, 163,0,127, 119,0,91, 63,0,43, 11,0,7],
AltRed: [255,0,0, 219,0,19, 183,0,35, 147,0,39, 115,0,43, 79,0,35, 43,0,23, 11,0,7],
AltYellow: [255,255,0, 231,231,0, 219,219,0, 199,199,0, 155,155,0, 115,115,0, 75,75,0, 35,35,0],
AltGreenblue: [0,255,211, 0,227,179, 0,199,151, 0,171,127, 0,135,103, 0,103,75, 0,55,39, 0,11,7],
AltPurple: [255,99,255, 215,67,223, 179,43,195, 147,23,167, 111,7,139, 83,0,111, 47,0,63, 11,0,15],
AltPurple2: [219,127,255, 203,63,255, 187,0,255, 163,0,211, 139,0,171, 107,0,127, 51,0,67, 7,0,11],
JustinYellow: [255,255,4, 200,200,0, 155,156,0, 130,130,0, 116,116,0, 63,62,0, 47,35,0, 11,11,0],
JustinBlue: [0,138,255, 0,107,204, 0,93,177, 0,79,151, 0,47,98, 0,39,71, 0,23,47, 7,6,12],
JustinBlue2: [7,251,248, 6,203,248, 0,152,239, 0,101,231, 0,48,224, 4,0,224, 0,23,148, 0,43,72],
JustinPink: [228,179,255, 219,126,255, 247,112,163, 241,51,131, 240,0,100, 199,0,0, 143,0,0, 63,0,0],
JustinOrange: [255,199,4, 254,148,4, 254,95,4, 255,2,0, 239,0,100, 191,0,76, 146,0,56, 79,0,11],
JustinBlack: [109,108,109, 63,63,62, 33,32,33, 16,16,16, 8,9,8, 7,7,7, 3,4,4, 2,3,3],
JustinMagenta: [236,64,127, 216,30,127, 193,0,127, 166,0,104, 141,0,85, 111,0,63, 56,0,33, 9,3,5],
MouseguyDeadscrap: [204,102,81, 178,63,55, 153,30,30, 127,12,22, 102,0,17, 76,0,19, 51,0,16, 25,0,10]
};
let recolor = new Uint8Array(3 * (10 * 8 + 1));
recolor[0] = recolor[1] = recolor[2] = 255; //white for Lori's eyes
let gradients = document.getElementById("gradients");
let addText = "<li><ul class='gradient'><li><select>";
addText += "<option>Fill</option>";
addText += "<option>1-color Gradient</option>";
addText += "<option>2-color Gradient</option>";
addText += "<option>3-color Gradient</option>";
addText += "<option>Fully Custom</option>";
for (let key in CanonGradients)
addText += "<option>" + key + "</option>";
addText += "</select></li>";
for (let c = 0; c < 8; ++c)
addText += "<li><input type='color' /></li>";
addText += "</ul></li>";
for (let i = 0; i < 10; ++i)
gradients.insertAdjacentHTML("beforeend", addText);
let selects = document.getElementsByTagName("select");
let rows = document.getElementsByClassName("gradient");
for (let i = 0; i < 10; ++i) {
let select = selects[i];
let children = rows[i].childNodes;
select.addEventListener("change", function(){
let enableds;
switch (this.selectedIndex) {
case 0: //fill
enableds = [true, false, false, false, false, false, false, false];
break;
case 1: //1-color gradient
enableds = [false, false, true, false, false, false, false, false];
break;
case 2: //2-color gradient
enableds = [false, false, true, false, false, true, false, false];
break;
case 3: //3-color gradient
enableds = [true, false, true, false, false, true, false, false];
break;
case 4: //fully custom
enableds = [true, true, true, true, true, true, true, true];
break;
default:
enableds = [false, false, false, false, false, false, false, false];
let channels = CanonGradients[this.value];
for (let c = 0; c < 8; ++c) {
const recolorOffset = (1 + c + i * 8) * 3;
for (let cc = 0; cc < 3; ++cc)
recolor[recolorOffset + cc] = channels[c * 3 + cc];
}
break;
}
for (let c = 0; c < 8; ++c)
children[c + 1].className = enableds[c] ? "enabled" : "";
generateGradient(i);
});
function generateGradient(gradientID) {
let recolorOffset = (1 + gradientID * 8) * 3;
const gradientType = selects[gradientID].selectedIndex;
if (gradientType == 0) { //fill
for (let c = 0; c < 24; c += 3)
for (let cc = 0; cc < 3; ++cc)
recolor[recolorOffset + c + cc] = recolor[recolorOffset + cc];
} else if (gradientType < 4) { //partially defined gradient
function multiplyColor(oldOffset, newOffset, factor) {
oldOffset *= 3;
newOffset *= 3;
for (let cc = 0; cc < 3; ++cc) {
let intensity = recolor[recolorOffset + oldOffset + cc];
intensity = intensity * factor | 0;
if (intensity) {
if (intensity > 255)
intensity = 255;
else
intensity |= 3;
}
recolor[recolorOffset + newOffset + cc] = intensity;
}
}
if (gradientType == 3) { //3-color gradient, defines top color
for (let cc = 0; cc < 3; ++cc)
recolor[recolorOffset + 3 + cc] = (recolor[recolorOffset + 0 + cc] + recolor[recolorOffset + 6 + cc]) / 2 | 3; //color 1 is average of colors 0 and 2
} else { //generate top color
multiplyColor(2, 1, 1.2);
multiplyColor(2, 0, 1.5);
}
if (gradientType == 1) { //1-color gradient, need to derive a lot of middle colors
multiplyColor(2, 3, 0.8);
multiplyColor(2, 4, 0.6);
multiplyColor(2, 5, 0.4);
} else { //colors 3 and 4 are averages of 2 and 5
for (let cc = 0; cc < 3; ++cc) {
const startColor = recolor[recolorOffset + 6 + cc];
const step = (recolor[recolorOffset + 15 + cc] - startColor) / 3;
recolor[recolorOffset + 9 + cc] = (startColor + step) | 3;
recolor[recolorOffset + 12 + cc] = (startColor + step * 2) | 3;
}
}
multiplyColor(5, 6, 0.525);
multiplyColor(5, 7, 0.125);
}
for (let c = 0; c < 8; ++c) {
const rgbstring = recolor[recolorOffset] + "," + recolor[recolorOffset + 1] + "," + recolor[recolorOffset + 2];
children[c + 1].title = rgbstring;
children[c + 1].style.background = children[c + 1].style.borderColor = "rgb(" + rgbstring + ")";
children[c + 1].childNodes[0].value = "#" + recolor[recolorOffset].toString(16).padStart(2, '0') + recolor[recolorOffset + 1].toString(16).padStart(2, '0') + recolor[recolorOffset + 2].toString(16).padStart(2, '0');
recolorOffset += 3;
}
}
select.selectedIndex = i + 5;
select.dispatchEvent(new Event("change"));
for (let cell = 1; cell < 9; ++cell) {
let li = children[cell];
li.childNodes[0].addEventListener("input", function() {
const r = parseInt(this.value.substr(1,2), 16);
const g = parseInt(this.value.substr(3,2), 16);
const b = parseInt(this.value.substr(5,2), 16);
const recolorOffset = (1 + i * 8 + (cell - 1)) * 3;
recolor[recolorOffset + 0] = r;
recolor[recolorOffset + 1] = g;
recolor[recolorOffset + 2] = b;
generateGradient(i);
draw();
});
}
}
const grayscale = new Image();
grayscale.src = "";
function draw() {
const cnvs = document.createElement("canvas");
const w = grayscale.width, h=grayscale.height;
if (!h) {
return setTimeout(draw, 500);
}
cnvs.width = w; cnvs.height = h;
const ctx = cnvs.getContext("2d");
ctx.drawImage(grayscale, 0,0);
const imgdata = ctx.getImageData(0,0,w,h);
const rgba = imgdata.data;
for (let px=0,ct=w*h*4; px<ct; px+=4) {
const brightness = rgba[px];
if (brightness) //not color 0, which is transparent
for (let ch = 0; ch < 3; ++ch) {
const recolorOffset = (brightness - 15) * 3;
rgba[px+ch] = recolor[recolorOffset + ch];
}
}
ctx.putImageData(imgdata,0,0);
document.getElementById("preview").style.backgroundImage = "url('" + cnvs.toDataURL() + "')";
}
draw();
for (let i = 0; i < 10; ++i)
selects[i].addEventListener("change", draw);
let saves = document.getElementsByClassName("butsave");
for (let i = 0; i < 4; ++i) {
saves[i].addEventListener("click", function(){
let output = new Uint8Array(1 + 10 + recolor.length);
output[0] = 0; //version
for (let i = 0; i < 10; ++i)
output[i + 1] = selects[i].selectedIndex;
output.set(recolor, 1 + 10);
//https://stackoverflow.com/questions/3665115/how-to-create-a-file-in-memory-for-user-to-download-but-not-through-server
const element = document.createElement('a');
element.setAttribute('href', window.URL.createObjectURL(new Blob([output])));
element.setAttribute('download', "Fur" + (i + 1) + ".asdat");
element.style.display = 'none';
document.body.appendChild(element);
element.click();
document.body.removeChild(element);
});
}
document.getElementById("butload").addEventListener("click", function() {
document.getElementById('hiddenload').click();
});
document.getElementById('hiddenload').addEventListener('change', function(evt) {
let f = evt.target.files[0];
if (f) {
if (f.name != "Fur1.asdat" && f.name != "Fur2.asdat" && f.name != "Fur3.asdat" && f.name != "Fur4.asdat") {
alert("Invalid filename");
return;
}
if (f.size != 1 + 10 + recolor.length) {
alert("Invalid filesize");
return;
}
const reader = new FileReader();
reader.onload = function() {
const result = new Uint8Array(reader.result);
let idx = 0;
if (result[idx++] > FILEVERSION) {
alert("File saved in a later version of TrueFur.html, please update your local copy.");
return;
}
for (let selectID = 0; selectID < 10; ++selectID)
selects[selectID].selectedIndex = result[idx++];
recolor = result.slice(idx);
for (let selectID = 0; selectID < 10; ++selectID)
selects[selectID].dispatchEvent(new Event("change"));
};
reader.readAsArrayBuffer(f);
}
}, false);
});
</script>
</body>
Jazz2Online © 1999-INFINITY (Site Credits). We have a Privacy Policy. Jazz Jackrabbit, Jazz Jackrabbit 2, Jazz Jackrabbit Advance and all related trademarks and media are ™ and © Epic Games. Lori Jackrabbit is © Dean Dodrill. J2O development powered by Loops of Fury and Chemical Beats.
Eat your lima beans, Johnny.