/**
* Color Sorting
* Sort the liquids so that each container only contains the same color. How fast can you sort them?
*
* How to use:
* - Click/tap a container to select it as the source container.
* - Click/tap it again to unselect it.
* - Click/tap another container to select it as destination.
* - The top liquid from the source container will be transfered onto the top of the destination container.
*
* Conversion table for the minified code:
* f = isAnimating
* g = stocks
* h = containers
* m = timeStart
* n = Date.now
* o = initialize
* p = containerChecker
* q = selection
* r = prevSelection
* s = size
* t = tileSize
* u = tileSize * 4 / 5
* v = 25
* w = draw
*/
size = Math.min(a.width, a.height); // Fit to window
a.width = size; // Make the board a square
a.height = size;
tileSize = size / 5;
timeStart = 0; // Timestamp of the first liquid transfer
selection = -1; // -1 = none
prevSelection = -1; // (same as above)
isAnimating = 0; // The offset when the liquid transfer is happening
stocks = []; // Used for liquid initialization
containers = []; // The current state of the containers
// Helper function to check whether a container is full and single-colored
containerChecker = k => {
if (k.length < 5) {
return false;
}
z = k[0];
for (j = 1; j < 5; j ++) {
if (k[j] != z) {
z = -1;
break;
}
}
return z > -1;
};
// Function to create a randomized board
initialize = e => {
stocks = [25, 25, 20, 25, 25]; // Green is less due to 1 empty container.
containers = [];
for (i = 25; i > 0; ) {
z = [];
if (i -- != 13) {
for (j = 5; j; j --) {
k = Math.floor(Math.random() * 4);
while (stocks[k] <= 0) {
// This ensures that there are no pairs of the same color.
k = Math.floor(k + 1 + Math.random() * 3) % 5;
}
z.push(k);
stocks[k] --;
}
}
containers.push(z);
}
i = 0;
for (; i < 25; i ++) {
if (containerChecker(containers[i])) {
break;
}
}
if (i < 25) {
initialize();
}
};
initialize();
// Function to draw the board
draw = e => {
for (y = 0; y < 5; y ++) {
for (x = 0; x < 5; x ++) {
i = x + y * 5;
if (prevSelection == i) { // The selected container
c.fillStyle = "#ffa";
} else if (containerChecker(containers[i])) { // The "correct" container
c.fillStyle = "#afa";
} else {
c.fillStyle = "#fff";
}
c.fillRect(tileSize * x, tileSize * y, tileSize, tileSize);
c.strokeRect(tileSize * (x + 0.1), tileSize * (y + 0.1), tileSize * 4 / 5, tileSize * 4 / 5);
k = containers[i].length;
j = 0;
z = 0;
for (; z < k; z ++) {
if (isAnimating && z == k - 1) {
if (prevSelection == i) { // The source container is giving the top liquid.
j = tileSize * 4 / 25 - isAnimating;
} else if (selection == i) { // The destination container is receiving a liquid on top.
j = isAnimating;
}
}
hue = containers[i][z] / 5 * 360;
c.fillStyle = `hsl(${hue},100%,50%)`;
c.fillRect(tileSize * (x + 0.1), tileSize * (y + 0.1 + 16 / 25 - z * 4 / 25) + j, tileSize * 4 / 5, tileSize * 4 / 25 - j);
}
}
}
if (isAnimating > 0) {
setTimeout(draw, 9);
isAnimating = Math.max(0, isAnimating - 2);
if (isAnimating == 0) {
// The top liquid is removed from the source container only after the animation has completed.
containers[prevSelection].pop();
// All selections are cleared after the transfer is done.
selection = prevSelection = -1;
}
}
};
draw();
// Function to be called when a click is happening
a.onclick = e => {
if (isAnimating > 0) {
// The board is frozen/disabled when an animation is still happening.
return;
}
selection = Math.floor(e.x / tileSize) + Math.floor(e.y / tileSize) * 5;
if (prevSelection < 0) {
if (containers[selection].length > 0) {
// The player can only select non-empty containers.
prevSelection = selection;
}
} else if (prevSelection == selection) {
// This allows player to unselect a container.
selection = -1;
prevSelection = -1;
} else if (containers[selection].length < 5) {
// The player can only select a non-full container as the destination.
containers[selection].push(containers[prevSelection].slice(-1)[0]);
isAnimating = tileSize * 4 / 25;
if (timeStart == 0) {
timeStart = Date.now();
}
}
draw();
x = 0;
i = 0;
for (; i < 25; ) {
if (containerChecker(containers[i ++])) {
x ++;
}
}
if (x > 23) {
// A winning alert is shown after the player has completed the task.
alert(`Won! (${(Date.now()-timeStart)/1000} s)`);
}
};