/* * Fun with Binary - a fun way of introducing binary * Copyright (C) 2018, Diogo Cordeiro. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * * @category UI interaction * @package Fun with Binary * @author Diogo Cordeiro * @copyright 2018 Diogo Cordeiro. * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 * @link https://www.diogo.site/projects/fun_with_binary/ */ /*********************************** * EXTEND JAVASCRIPT FUNCTIONALITY * ***********************************/ /** * Returns a random integer between min(inclusive) and max(inclusive) * Using Math.round() would give a non-uniform distribution! */ function getRandomInt(min, max) { return Math.floor(Math.random() *(max - min + 1)) + min; } /** * Replace char at given position */ String.prototype.replaceAt=function(index, replacement) { return this.substr(0, index) + replacement+ this.substr(index + replacement.length); } /** * Checks if session storage is available */ function is_session_storage_available() { return window.sessionStorage != null; } /*********************************** * GAME SETTINGS * ***********************************/ // Online Mode (only used in Online Mode) ONLINEMODE = true; // Box IP ONLINEURL = "http://42.42.42.42/game/"; // Number of leds NLEDS = 6; // Activate the Show 2 powers Labels feature HAVELABELS = true; // Default Show 2 powers Labels feature mode show_labels = true; // Easter Egg EASTEREGG = true; // ON and OFF leds srcs LED_OFF = ""; LED_ON = ""; /********************************** * ACTUAL GAME CODE * **********************************/ /** * Global variables */ // Empty Answer empty_answer = ''; while (empty_answer.length < NLEDS) { empty_answer += '0'; } // correct answer correct_answer = empty_answer; // current answer current_answer = empty_answer; // Generate a token for this session sessionStorage.token = Math.random().toString(36).substr(2, 15+2); /** * Create leds */ for (let i = 0; i < NLEDS; ++i) { // Led holder let figure = document.createElement("figure"); figure.setAttribute("class", "box"); // Label if (HAVELABELS) { let c_label = Math.pow(2, NLEDS-i-1); let figcaption = document.createElement("figcaption"); figcaption.setAttribute("value", c_label); figcaption.setAttribute("class", "text labels"); figcaption.id = "label" + i; figcaption.innerHTML = c_label; figure.appendChild(figcaption); } else { document.getElementById('label_switch_button').setAttribute("style", "display:none"); } // Led let img = document.createElement("img"); // Set img src if (is_session_storage_available() && sessionStorage.playing == 'true' && sessionStorage.current_answer.charAt(i) == '1') { img.src = LED_ON; img.setAttribute("value", "ON"); } else { img.src = LED_OFF; img.setAttribute("value", "OFF"); } img.id = "led" + i; img.setAttribute("class", "led"); img.setAttribute("onclick", "switch_led("+i+")"); figure.appendChild(img); document.getElementById("leds").appendChild(figure); } // If initial value for label is false, handle it if (HAVELABELS && !show_labels) { show_labels = true; document.getElementById("label_switch").click(); } // Let the game begin get_game(); /** * Switch labels */ function switch_labels() { if (show_labels) { for (let i = 0; i < NLEDS; ++i) { document.getElementById("label"+i).setAttribute ("style", "visibility:hidden"); } } else { for (let i = 0; i < NLEDS; ++i) { document.getElementById("label"+i).setAttribute ("style", ""); } } show_labels = !show_labels; cache_current_game(); } /** * Turn current answer bits accordingly */ function switch_led(led_id) { let image = document.getElementById("led"+led_id); if (image.getAttribute("value") == "ON") { image.setAttribute("value", "OFF"); image.src = LED_OFF; current_answer = current_answer.replaceAt(led_id, "0"); } else { image.setAttribute("value", "ON"); image.src = LED_ON; current_answer = current_answer.replaceAt(led_id, "1"); } cache_current_game(); // Easter egg if (EASTEREGG && current_answer == (empty_answer.substr(6) + "101010")) { alert("That's the answer to life, the universe and everything!"); } // If we are connected to the box if (ONLINEMODE) { let xhttp = new XMLHttpRequest(); xhttp.open("GET", ONLINEURL+"switch_state?led=" + led_id + "&token=" + sessionStorage.token, true); xhttp.send(); } // If right answer, finish game, restart a new one if (current_answer == correct_answer) { end_game(); restart_game(); } } /** * Sets problem statement's decimal number */ function set_statement(n_dec) { document.getElementById("statement_value").innerHTML = n_dec; } /** * Recovers game from page reload */ function get_game() { if (!is_session_storage_available() || sessionStorage.playing != 'true') { start_game(); } else { restore_from_cache(); } } /** * Restore current game from cache */ function restore_from_cache() { correct_answer = sessionStorage.correct_answer; current_answer = sessionStorage.current_answer; set_statement(Number(sessionStorage.number_dec)); if (HAVELABELS && sessionStorage.show_label != show_labels.toString()) { show_labels = true; document.getElementById("label_switch").click(); } } /** * Caches current game */ function cache_current_game() { if (!is_session_storage_available()) { return; } sessionStorage.current_answer = current_answer; sessionStorage.show_label = show_labels; } /** * Starts game */ function start_game() { let n_dec = getRandomInt(1, Math.pow(2, NLEDS)-1); let n_bin = n_dec.toString(2); correct_answer = empty_answer.substr(n_bin.length) + n_bin; set_statement(n_dec); if (is_session_storage_available()) { sessionStorage.playing = true; sessionStorage.correct_answer = correct_answer; sessionStorage.number_dec = n_dec; cache_current_game(); } } /** * Prepares everything for a new game */ function reset_game() { correct_answer = empty_answer; current_answer = empty_answer; turn_all_leds_images_off(); for (let i = 0; i < NLEDS; ++i) { document.getElementById("led"+i).setAttribute("value", "OFF"); } } /** * Start a new game */ function restart_game() { reset_game(); start_game(); } /** * Finishes the game */ function end_game() { // If we are connected to the box if (ONLINEMODE) { let xhttp = new XMLHttpRequest(); xhttp.open("GET", ONLINEURL+"won" + "?token=" + sessionStorage.token, true); xhttp.send(); } // Blink correct answer leds 3 times blink_led_effect(3, correct_answer); // Reset cache if (is_session_storage_available()) { sessionStorage.clear(); } } /** * Turn every led on(without turning value attribute) */ function turn_leds_images_on(which) { // Turn all on for (let i = 0; i < NLEDS; ++i) { if (which.charAt(i) == '1') { document.getElementById("led"+i).src = LED_ON; } } } /** * Turn every led off(without turning value attribute) */ function turn_all_leds_images_off() { // Turn all off for (let i = 0; i < NLEDS; ++i) { document.getElementById("led"+i).src = LED_OFF; } } /** * Makes leds blink i times */ function blink_led_effect(i, which) { // Because the recursive function fulfills an action every two iterations blink_led_effect_helper(i*2, which); } /** * Makes leds blink i/2 times */ function blink_led_effect_helper(i, which) { if (i % 2 == 1) { turn_leds_images_on(which); } else { turn_all_leds_images_off(); } if (--i > -1) { setTimeout(function() { blink_led_effect_helper(i, which); }, 500); } }