Flood-It game in JavaScript

Turn: /

This is a JavaScript game consisting of trying to flood fill the entire box with one colour using twenty-five or fewer "flood fills". It's based on floodit.appspot.com, which I found via How to optimally solve the flood fill puzzle? However, this is a from-scratch reimplementation.

There is also version without the programming details here: Play the game

flood-game.js

This is the JavaScript for the game:
var n_rows = 14;
var n_cols = 14;
var start_table = new Array (n_rows);
for (var row = 0; row < n_rows; row++) {
    start_table[row] = new Array (n_cols);
}
var colours = "blue red green yellow pink purple".split (/\s+/);

var userReact = true;
/* DOM functions. */

function create_node (type, parent)
{
    var new_node = document.createElement (type);
    parent.appendChild (new_node);
    return new_node;
}

function append_text (parent, text)
{
    var text_node = document.createTextNode (text);
    clear (parent);
    parent.appendChild(text_node);
}

function getByID (id)
{
    var element = document.getElementById (id);
    return element;
}

function clear (element)
{
    while (element.lastChild)
        element.removeChild (element.lastChild);
}

/* Flood fill game. */

var moves;
var max_moves = 25;
var finished;
var nFlooded = 0;

/* Alter a flooded element's colour. */
function colourFlooded(row, col, colour)
{
    game_table[row][col].colour = colour;
    game_table[row][col].element.className = "piece " + colour;
}

/* Mark a square as not flooded. This is for the undo function. */
function unfloodSq(row, col) {
    game_table[row][col].flooded = false;
    game_table[row][col].move = unmovedState;
}

/* Mark a square as flooded, and record the move number in the square
   so that the move can be undone. */
function floodSq(row, col, move) {
    game_table[row][col].flooded = true;
    game_table[row][col].move = move;
}

/* The square specified by row and col is adjacent to a flooded
   square, so see if we can flood this one too. */
function floodAdjacent(row, col, move) {
    if (game_table[row][col].flooded)
        return;
    if (game_table[row][col].colour == move.colour) {
        /* This is the right colour, so flood it. */
        nFlooded++;
        floodSq(row, col, move.move);
        /* Recurse to get connected neighbours. */
        floodNeighbours(row, col, move);
    }
}

/* Given a square at row, col which is flooded, look at all its
   neighbours to see if they can be flooded too. */
function floodNeighbours(row, col, move)
{
    if (row < n_rows - 1)
        floodAdjacent(row + 1, col, move);
    if (row > 0)
        floodAdjacent(row - 1, col, move);
    if (col < n_cols - 1)
        floodAdjacent(row, col + 1, move);
    if (col > 0)
        floodAdjacent(row, col - 1, move);
}

/* This returns true if the entire board is flooded, false if not. */
function all_flooded ()
{
    for (var row = 0; row < n_rows; row++) {
        for (var col = 0; col < n_cols; col++) {
            if (! game_table[row][col].flooded) {
                return false;
            }
        }
    }
    return true;
}

/* Disable one of the colour choice buttons. */
function disableChosen(colour) {
    var allbuttons = document.getElementsByClassName("button")
    for (var i = 0; i < allbuttons.length; i++) {
        if (allbuttons[i].classList.contains(colour)) {
            allbuttons[i].style = "opacity: 0%;"
        } else {
            allbuttons[i].style = "opacity: 100%;";
        }
    }
}

/* Choices made by the user. This is for the undo function. */
var choices = new Array();

/* Save the move so that it can be undone. */
function saveMove(move) {
    choices.push(move);
}

/* Do "action" on all the squares of game_table. */
function allSq(action) {
    for (var row = 0; row < n_rows; row++) {
        for (var col = 0; col < n_cols; col++) {
            action(row, col);
        }
    }
}

/* Flood the board with a new colour specified by move.colour. */
function flood(move, initial)
{
    if (finished)
        return;
    var old_colour = game_table[0][0].colour;
    /* Check this is not the same as the old colour. */
    if (! initial && move.colour == old_colour)
        return;
    /* Check how many squares have changed. */
    nFlooded = 0;
    /* Change the colour of all the already-flooded elements. */
    allSq(function(row, col) {
        if (game_table[row][col].flooded) {
            colourFlooded (row, col, move.colour);
            floodNeighbours(row, col, move);
        }
    });
    clearStatus();
    /* In the initial case only, it's OK if nothing has changed. */
    if (initial || nFlooded > 0) {
        moves++;
        saveMove(move);
        if (! initial) {
            react();
        }
    } else {
        /* Ignore this move as it was useless. */
        useless();
    }
    if (all_flooded()) {
        finished = true;
        if (moves <= max_moves) {
            winner();
        } else {
            setStatus("🥱 Finished, at last! 😴");
        }
    } else if (moves >= max_moves) {
        setStatus("😞 You lost! 💔");
    }
    updateTurn();
    disableChosen(move.colour);
}

function remainingColours() {
    var remaining = new Object();
    allSq(function(row, col) {
        var sq = game_table[row][col];
        if (! sq.flooded) {
            remaining[sq.colour] = true;
        }
    });
    return Object.keys(remaining).length;
}

function rando(arr) {
    var r = Math.floor(Math.random()*arr.length);
    setStatus(arr[r])
}

var notLookingGood = [
    "⚕️ Prognosis: negative 😷",
    "Things are looking gloomy!",
    "There may be trouble ahead...",
    "It's not whether you win or lose, it's how you play the game.",
    "Il y a beaucoup de dangers!",
];

var previousRemaining = 6;

/* React to the user's play. */
function react() {
    if (! userReact) {
        return
    }
    var remaining = remainingColours();
    var movesLeft = max_moves - moves;
    if (remaining > movesLeft) {
        rando(notLookingGood);
        return;
    }
    if (remaining < previousRemaining) {
        setStatus("All of the " + game_table[0][0].colour + " squares have been flooded");
        previousRemaining = remaining;
        return;
    }
    /* I don't think it's possible to get more than 30 at once. */
    if (nFlooded > 25) {
        setStatus("🤯 Mind Boggling! 🤯");
    } else if (nFlooded > 20) {
        setStatus("🐶 Wowzer! 🐱");
    } else if (nFlooded > 15) {
        setStatus("🚀 Star player! ⭐");
    } else if (nFlooded > 10) {
        setStatus("🔥 On fire!");
    } else if (nFlooded > 6) {
        setStatus("😍 Great!");
    } else if (nFlooded > 4) {
        setStatus("😎 Nice!");
    } else if (nFlooded > 2) {
        setStatus("👍");
    } else if (nFlooded == 1) {
        if (moves > 2) {
            setStatus("🥱");
        }
    }       
}

/* Update the "Turn" counter. */
function updateTurn() {
    append_text (getByID ("moves"), moves);
}

/* Publish a congratulatory message indicating successful
   completion. */
function winner() {
    var winner = [
        "🎉 You win! 🥂",
        "🎈 Congratulations! 🎈",
        "🍗 Winner, winner, chicken dinner! 🍗",
        "🏆 An overwhelming victory! 🏆",
        "🎯 You have nailed it! 🎯",
        "🍒 You are the champion, my friend! 🍈",
    ];
    var r = max_moves - moves;
    if (r >= winner.length) {
        r = winner.length - 1;
    }
    setStatus(winner[r]);
}

var nUseless = 0;

/* Make a silly response when the user keeps clicking invalid
   choices. */
function useless() {
    nUseless++;
    if (nUseless < 5) {
        setStatus("There were no adjacent squares with that colour!");
        return;
    }
    rando(suggest);
}

    var suggest = [
        "🍜 Use your noodle",
        "🍄 Better lay off the mushrooms",
        "🎓 Put on your thinking cap",
        "🐻 You may not be smarter than the average bear",
        "🗿 The secret is to bang the rocks together, guys",
    ];


var statusEl

function status() {
    statusEl = getByID("status");
    return statusEl;
}

function clearStatus() {
    if (! userReact) {
        return;
    }
    clear(status());
}

function setStatus(msg) {
    if (! userReact) {
        return
    }
    var s = status();
    clear(s);
    s.innerHTML = msg;
}

function help ()
{
    setStatus ("Press the circle buttons to flood fill the image\n"+
               "with the colour from the top left corner. Fill the\n"+
               "entire image with the same colour in twenty-five or\n"+
               "fewer flood fills.");
}

/* Create a random colour for "createTable". */
function random_colour ()
{
    var colour_no = Math.floor (Math.random () * 6);
    return colours[colour_no];
}

/* The "state of play" is stored in game_table. */
var game_table = new Array (n_rows);
for (var row = 0; row < n_rows; row++) {
    game_table[row] = new Array (n_cols);
    for (var col = 0; col < n_cols; col++) {
        game_table[row][col] = new Object ();
    }
}

/* This square has not had anything done to it. */
const unmovedState = -1;
/* This square is part of the "initial state", it is either the top
   left square on starting the game, or a square connected to that and
   with the same colour. */
const initialState = 0;

/* Create the initial table. Set ran to true to make a random table,
   else the table is loaded from the existing values of game_table. */
function createTable(ran)
{
    moves = -1;
    finished = false;
    var game_table_element = getByID ("game-table-tbody");
    clear(game_table_element);
    for (var row = 0; row < n_rows; row++) {
        var tr = create_node ("tr", game_table_element);
        for (var col = 0; col < n_cols; col++) {
            var td = create_node ("td", tr);
            if (ran) {
                var colour = random_colour ();
                game_table[row][col].colour = colour;
            }
            td.className = "piece " + game_table[row][col].colour;
            unfloodSq(row, col);
            start_table[row][col] = colour;
            game_table[row][col].element = td;
        }
    }
    /* Mark the first element of the table as flooded. */
    floodSq(0, 0, initialState);
    var initialColour = game_table[0][0].colour;
    /* Initialize the adjacent elements with the same colour to be flooded
       from the outset. */
    flood({colour: initialColour, move: initialState}, true);
    append_text(getByID("max-moves"), max_moves);
    if (ran) {
        /* Save the initial board. */
        storeInitial();
    }
}

/* Click function for the colour buttons. */
function choose(colour) {
    flood({colour: colour, move: moves + 1});
}

/* Reset the board and variables. */
function resetBoard() {
    previousRemaining = 6;
    choices = new Array();
}

/* Start a new game. */
function newGame ()
{
    nUseless = 0;
    resetBoard();
    clearSave();
    squashed = "";
    createTable(true);
}

function clearSave() {
    var url = document.location.href;
    var parts = url.split('?');
    url = parts.shift();
    window.history.pushState('', document.title, url);
}

function sleep(delay) {
    return new Promise(resolve => setTimeout(resolve, delay));
}

/* Show the user the solver's solution, with a delay between each
   move. */
async function solution(response) {
    var sol = JSON.parse(response)
    if (! sol.ok) {
        setStatus("Solver complained as follows: " + sol.status)
        return
    }
    if (! sol.solved) {
        setStatus("The computer was not smart enough to solve this!")
        return
    }
    setDifficulty(sol.difficulty)
    userReact = false
    for (var i = 0; i < sol.choices.length; i++) {
        flood({colour: sol.choices[i], move: i+1}, false)
        await sleep(500);
    }
    userReact = true
}

function setDifficulty(d) {
    switch (d) {
    case 0:
        setStatus("🥱 Very easy!");
        return
    case 1:
        setStatus("☺️ Easy peasy!");
        return
    case 2:
        setStatus("😐 Quite difficult!");
        return
    case 3:
        setStatus("😕 Difficult!");
        return
    case 4:
        setStatus("😟 Very difficult!");
        return
    case 5:
        setStatus("😖 Extremely difficult!");
        return
    }
}


/* Send this board to the solver. */
function solve() {
    if (finished) {
        setStatus("The game is already solved!");
        return;
    }
    var board = new Array(n_rows);
    for (var i = 0; i < n_rows; i++) {
        board[i] = new Array(n_cols);
        for (var j = 0; j < n_cols; j++) {
            var sq = game_table[i][j]
            board[i][j] = {
                c: sq.colour,
                f: sq.flooded,
            }
        }
    }
    var game = {
        moves: moves,
        board: board,
    };
    var sender = new XMLHttpRequest();
    sender.onreadystatechange = function() {
        if (this.readyState == 4) {
            if (this.status == 200) {
                solution(sender.responseText);
            } else {
                setStatus("Failed to get a solution, you'll have to do it yourself!");
            }
        }
    }
    sender.open("POST", "/floodsolver/", true);
    sender.send(JSON.stringify(game));
}

function startAgain() {
    if (moves == 0) {
        setStatus("We are at the start already");
        return;
    }
    resetBoard();
    unsquash(squashed);
    createTable(false);
}

/* Mutex for undoing. User hitting the undo button repeatedly can
   cause weird problems. */
var undoing = false;

/* Undo the most recent move. */
function undo() {
    if (undoing) {
        return;
    }
    if (moves <= 0) {
        setStatus("There is nothing to undo, pick a colour first.");
        return;
    }
    undoing = true;
    if (finished) {
        finished = false;
    }
    clearStatus();
    var lastMove = choices[choices.length - 1]
    var prevMove = choices[choices.length - 2]
    allSq(function(row, col) {
        var sq = game_table[row][col];
        if (sq.flooded) {
            if (sq.move == moves) {
                unfloodSq(row, col);
            } else {
                colourFlooded(row, col, prevMove.colour);
            }
        }
    });
    moves--;
    updateTurn();
    choices.pop();
    disableChosen(prevMove.colour);
    previousRemaining = remainingColours();
    undoing = false;
}

function colourToNumber(colour) {
    switch (colour) {
    case "blue":
        return 0;
    case "red":
        return 1;
    case "green":
        return 2;
    case "yellow":
        return 3;
    case "pink":
        return 4;
    case "purple":
        return 5;
    }
    throw("Unknown colour "+colour);
}

function numberToColour(num) {
    if (num >= colours.length) {
        throw("Number too big " + num);
    }
    return colours[num];
}

/* Convert n to letters/digits/URL-safe characters. */
function sixtysix(n) {
    if (n < 10) {
        return '0123456789'.charAt(n);
    }
    if (n < 36) {
        return 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'.charAt(n - 10);
    }
    if (n < 62) {
        return 'abcdefghijklmnopqrstuvwxyz'.charAt(n - 36);
    }
    /* No, I do not care about putting the following in ASCII
       order. */
    switch (n) {
    case 62:
        return '-';
    case 63:
        return '.';
    case 64:
        return '_';
    case 65:
        return '~';
    }
}

function sqToNum(n) {
    var row = Math.floor(n / n_rows);
    var col = Math.floor(n % n_rows);
    if (col >= n_cols || row >= n_rows) {
        throw("Number out of bounds " + n);
    }
    var cname = game_table[row][col].colour;
    return colourToNumber(cname);
}

/* The initial state of play. */
var squashed = "";

/* Squash seven cells into three bytes of web-safe ASCII using
   6^7/66^3 ~=~ 1. This code assumes that n_cols is a multiple of
   seven. */
function storeInitial() {
    var n = n_rows * n_cols;
    var sevens = Math.floor(n / 7);
    squashed = ""
    for (var i = 0; i < sevens; i++) {
        var out = 0;
        for (var j = 0; j < 7; j++) {
            out = sqToNum(i*7+j) + 6*out;
        }
        if (out > 279936) { // 6^7
            throw("Out is too big "+out);
        }
        squashed = squashed.concat(sixtysix(out % 66));
        out = Math.floor(out/66);
        squashed = squashed.concat(sixtysix(out % 66)); 
        out = Math.floor(out/66);
        squashed = squashed.concat(sixtysix(out));
    }
}

function moveList() {
    var nc = choices.length
    var ml = "";
    for (var i = 0; i < nc; i++) {
        var ci = choices[i].colour
        if (ci == "purple") {
            ci = "l";
        } else {
            ci = ci.substring(0, 1);
        }
        ml += ci;
    }
    return ml;
}

function squash() {
    var url = '?b=' + squashed;
    if (moves > 0) {
        ml = moveList();
        url += "&m=" + ml;
    }
    window.history.pushState({}, '', url);
}

function setSq(n, colour) {
    var row = Math.floor(n / n_rows);
    var col = Math.floor(n % n_rows);
    if (col >= n_cols || row >= n_rows) {
        throw("Number out of bounds " + n);
    }
    var cname = numberToColour(colour);
    game_table[row][col].colour = cname;
}

function un66(q) {
    var c = q.charCodeAt(0);
    if (c >= 48 && c <= 57) {
        return c - 48;
    }
    if (c >= 65 && c <= 90) {
        return c - 65 + 10;
    }
    if (c >= 97 && c <= 122) {
        return c - 97 + 36;
    }
    switch (c) {
    case 45:
        return 62;
    case 46:
        return 63;
    case 95:
        return 64;
    case 126:
        return 65;
    }
    throw("Unknown char value " + c);
}

function unsquash(squashed) {
    var n = squashed.length;
    if (n != n_rows * n_cols * 3 / 7) {
        return false;
    }
    var threes = Math.floor(n / 3);
    for (var i = 0; i < threes; i++) {
        var t = i * 3;
        var a = un66(squashed.charAt(t))
        var b = un66(squashed.charAt(t + 1))
        var c = un66(squashed.charAt(t + 2))
        var cell = a + 66 * (b + 66 * c);
        for (var j = 0; j < 7; j++) {
            var o = i*7+(6-j);
            setSq(o, Math.floor(cell % 6));
            cell = Math.floor(cell / 6);
        }
    }
    return true;
}

function save() {
    squash();
    setStatus("Copy or bookmark the browser URL to reload this game board");
}

function letterToMove(letter) {
    switch (letter) {
        case "b": return "blue";
        case "g": return "green";
        case "p": return "pink";
        case "l": return "purple";
        case "r": return "red";
        case "y": return "yellow";
    }
    throw("Unknown letter code "+letter);
}

function doMoves(ml) {
    var n = ml.length
    for (var i = 0; i < n; i++) {
        var move = ml.substring(i, i+1);
        var colour = letterToMove(move);
        flood({colour: colour, move: moves + 1});
    }
}

function load() {
    var regex = new RegExp("b=([0-9a-zA-Z_~.-]+)");
    // ~ is not web safe apparently, or is it?
    var results = regex.exec(decodeURI(window.location.href));
    if (results === null) {
        createTable(true);
        return;
    }
    squashed = results[1];
    if (! unsquash(squashed)) {
        setStatus("The starting position '" + squashed + "' looks wrong.");
        return;
    }
    createTable(false);
    var moveregex = new RegExp("m=([bglpry]+)");
    var moveresults = moveregex.exec(window.location.href);
    if (moveresults != null) {
        doMoves(moveresults[1]);
    }
}


You also need some styles and HTML to make this work.

game-table.html

This is the HTML for the game table. Note that the actual board is dynamically created in JavaScript.
<div id="game-board">
  <div id="colour-buttons">
    <table id="button-table">
      <tr>
        <td class="button blue" onclick="choose('blue')">
        </td>
        <td class="button red" onclick="choose('red')">
        </td>
        <td class="button green" onclick="choose('green')">
        </td>
      </tr>
      <tr>
        <td class="button yellow" onclick="choose('yellow')">
        </td>
        <td class="button pink" onclick="choose('pink')">
        </td>
        <td class="button purple" onclick="choose('purple')">
        </td>
      </tr>
    </table>
  </div> <!-- colour-buttons -->
  <div id="flood-controls">
    <table>
      <tr valign="top">
        <td id="turn-counter" valign="top">
          Turn: <span id="moves"></span>/<span id="max-moves"></span>
        </td>
      </tr>
      <tr valign="top">
        <td valign="top">
          <input type="submit" value="New game" onclick="newGame()">
        </td>
      </tr>
      <tr valign="top">
        <td valign="top">
          <input type="submit" value="Help" onclick="help()">
          <input type="submit" value="Undo" onclick="undo()">
        </td>
      </tr>
      <tr valign="top">
        <td valign="top">
          <input type="submit" value="Start again" onclick="startAgain()">
        </td>
      </tr>
      <tr valign="top">
        <td valign="top">
          <input type="submit" value="Solve" onclick="solve()">
          <input type="submit" value="Save" onclick="save()">
        </td>
      </tr>
    </table>
  </div>  <!-- flood-controls -->
  <div id="status">
  </div> <!-- status -->
  <table id="game-table">
    <tbody id="game-table-tbody">
    </tbody>
  </table>
</div> <!-- game-board -->

flood-game.css

These styles make the box colours and the table layout.
.red {
    background-color: red;
}

.yellow {
    background-color: gold;
}

.green {
    background-color: #8A865D;
}

.blue {
    background-color: blue;
}

.purple {
    background-color: purple;
}

.pink {
    background-color: #e77254;
}

#game-table {
    margin: 0px;
    border: 0px;
    padding: 0px;
    border-spacing: 0px;
    width: 392;
    height: 448;
    background: goldenrod;
    table-layout: fixed;
}

#status {
    width: 120px;
    height: 228px;
}

.piece {
    width: 28px;
    height: 30px;
    border: 0px;
    padding: 0px;
    margin: 0px;
}

.button {
    width: 30px;
    height: 30px;
    border: 0px;
    border-radius: 15px;
}

#button-table {
    border-spacing: 4px;
    vertical-align: top;
}

#turn-counter {
    font-family: sans-serif;
}

@media screen and (min-width: 600px) {
    #game-board {
        display: grid;
        grid-template-columns: 120px max-content;
        grid-template-rows: 80px 140px max-content;
    }

    #game-table {
        grid-column-start: 2;
        grid-column-end: 3;
        grid-row-start: 1;
        grid-row-end: 4;
    }

    #colour-buttons {
        grid-column-start: 1;
        grid-column-end: 2;
        grid-row-start: 1;
    }

    #flood-controls {
        grid-column-start: 1;
        grid-column-end: 2;
        grid-row-start: 2;
        grid-row-end: 3;
    }

    #status {
        grid-column-start: 1;
        grid-column-end: 2;
        grid-row-start: 3;
        grid-row-end: 4;
    }
}

@media screen and (max-width: 600px) {
    #status {
        max-width: 100px;
    }

    #game-board {
        display: grid;
        grid-template-columns: 24vmin 1fr 1fr;
        gap: 10px;
        margin: auto;
    }

    #game-table {
        grid-column-start: 1;
        grid-column-end: 4;
        grid-row-start: 1;
        width: 94.5vmin;
        height: 94.5vmin;
    }

    #colour-buttons {
        grid-column-start: 1;
        grid-row-start: 2;
    }

    #flood-controls {
        grid-column-start: 2;
        grid-column-end: 4;
        grid-row-start: 2;
    }

    #status {
        grid-column-start: 3;
        grid-row-start: 2;
    }

    .button {
        width: 6vmin;
        height: 6vmin;
        border-radius: 3vmin;
    }

    .piece {
        width: 6.75vmin;
        height: 6.75vmin;
    }
}


Copyright © Ben Bullock 2009-2024. All rights reserved. For comments, questions, and corrections, please email Ben Bullock (benkasminbullock@gmail.com) or use the discussion group at Google Groups. / Privacy / Disclaimer