mirror of https://github.com/espruino/BangleApps
Add "Bordle" app
parent
fff158b7c0
commit
288f2f4753
|
@ -0,0 +1,17 @@
|
||||||
|
# Bordle
|
||||||
|
|
||||||
|
The Bangle version of a popular word guessing game. The goal is to guess a 5 letter word in 6 tries or less. After each guess, the letters in the guess are
|
||||||
|
marked in colors: yellow for a letter that appears in the to-be-guessed word, but in a different location and green for a letter in the correct position.
|
||||||
|
|
||||||
|
Only words contained in the internal dictionary are allowed as valid guesses. On startup, a target word is picked from the dictionary at random.
|
||||||
|
|
||||||
|
On startup, a grid of 6 lines with 5 (empty) letter boxes is displayed. Swiping left or right at any time switches between grid view and keyboard view.
|
||||||
|
The keyboad was inspired by the 'Scribble' app (it is a simplified version using the layout library). The letter group "Z ..." contains the delete key ("<del>") and
|
||||||
|
the enter key ("<ent>"). Hitting enter after the 5th letter will add the guess to the grid view and color mark it.
|
||||||
|
|
||||||
|
The (English language) dictionary was derived from the the Unix ispell word list by filtering out plurals and past particples (and some hand editing) from all 5 letter words.
|
||||||
|
It is contained in the file 'wordlencr.txt' which contains one long string (no newline characters) of all the words concatenated. It would not be too difficult to swap it
|
||||||
|
out for a different language version. The keyboard currently only supports the 26 characters of the latin alphabet (no accents or umlauts).
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
require("heatshrink").decompress(atob("mEwxH+AA/TADwoIFkYyOF0owIF04wGUSqvVBZQtZGJYJIFzomKF0onIF07EKF0owLF9wNEnwACE6oZILxovbMBov/F/4v/C54uWF/4vKBQQLLF/4YPFwYMLF7AZGF5Y5KF5xJIFwoMJD44vaBhwvcLQpgHF8gGRF6xYNBpQvTXBoNOF65QJBIgvjBywvUV5YOOF64OIB54v/cQwAKB5ov/F84wKADYuIF+AwkFIwwnE45hmExCSlEpTEiERr3KADw+PF0ownUSoseA=="))
|
Binary file not shown.
After Width: | Height: | Size: 1.9 KiB |
|
@ -0,0 +1,159 @@
|
||||||
|
var Layout = require("Layout");
|
||||||
|
|
||||||
|
var gameState = 0;
|
||||||
|
var keyState = 0;
|
||||||
|
var keyStateIdx = 0;
|
||||||
|
|
||||||
|
function buttonPushed(b) {
|
||||||
|
if (keyState==0) {
|
||||||
|
keyState++;
|
||||||
|
keyStateIdx = b;
|
||||||
|
if (b<6) {
|
||||||
|
for (i=1; i<=5; ++i) {
|
||||||
|
var c = String.fromCharCode(i+64+(b-1)*5);
|
||||||
|
layout["bt"+i.toString()].label = c;
|
||||||
|
layout["bt"+i.toString()].bgCol = wordle.keyColors[c]||g.theme.bg;
|
||||||
|
}
|
||||||
|
layout.bt6.label = "<";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
layout.bt1.label = "Z";
|
||||||
|
layout.bt1.bgCol = wordle.keyColors.Z||g.theme.bg;
|
||||||
|
layout.bt2.label = "<del>";
|
||||||
|
layout.bt4.label = "<ent>";
|
||||||
|
layout.bt3.label = layout.bt5.label = " ";
|
||||||
|
layout.bt6.label = "<";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else { // actual button pushed
|
||||||
|
inp = layout.input.label;
|
||||||
|
if (b!=6) {
|
||||||
|
if ((keyStateIdx<=5 || b<=1) && inp.length<5) inp += String.fromCharCode(b+(keyStateIdx-1)*5+64);
|
||||||
|
else if (layout.input.label.length>0 && b==2) inp = inp.slice(0,-1);
|
||||||
|
layout.input.label = inp;
|
||||||
|
}
|
||||||
|
layout = getKeyLayout(inp);
|
||||||
|
keyState = 0;
|
||||||
|
if (inp.length==5 && keyStateIdx==6 && b==4) {
|
||||||
|
rc = wordle.addGuess(inp);
|
||||||
|
layout.input.label = "";
|
||||||
|
layout.update();
|
||||||
|
gameState = 0;
|
||||||
|
if (rc>0) return;
|
||||||
|
g.clear();
|
||||||
|
wordle.render();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
layout.update();
|
||||||
|
g.clear();
|
||||||
|
layout.render();
|
||||||
|
}
|
||||||
|
|
||||||
|
function getKeyLayout(text) {
|
||||||
|
return new Layout( {
|
||||||
|
type: "v", c: [
|
||||||
|
{type:"txt", font:"6x8:2", id:"input", label:text, pad: 3},
|
||||||
|
{type: "h", c: [
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt1", label:"ABCDE", cb: l=>buttonPushed(1), pad:4, filly:1, fillx:1 },
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt2", label:"FGHIJ", cb: l=>buttonPushed(2), pad:4, filly:1, fillx:1 },
|
||||||
|
]},
|
||||||
|
{type: "h", c: [
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt3", label:"KLMNO", cb: l=>buttonPushed(3), pad:4, filly:1, fillx:1 },
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt4", label:"PQRST", cb: l=>buttonPushed(4), pad:4, filly:1, fillx:1 },
|
||||||
|
]},
|
||||||
|
{type: "h", c: [
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt5", label:"UVWXY", cb: l=>buttonPushed(5), pad:4, filly:1, fillx:1 },
|
||||||
|
{type:"btn", font:"6x8:2", id:"bt6", label:"Z ...", cb: l=>buttonPushed(6), pad:4, filly:1, fillx:1 },
|
||||||
|
]}
|
||||||
|
]});
|
||||||
|
}
|
||||||
|
|
||||||
|
class Wordle {
|
||||||
|
constructor(word) {
|
||||||
|
this.word = word;
|
||||||
|
this.guesses = [];
|
||||||
|
this.guessColors = [];
|
||||||
|
this.keyColors = [];
|
||||||
|
this.nGuesses = -1;
|
||||||
|
if (word == "rnd") {
|
||||||
|
this.words = require("Storage").read("wordlencr.txt");
|
||||||
|
i = Math.floor(Math.floor(this.words.length/5)*Math.random())*5;
|
||||||
|
this.word = this.words.slice(i, i+5).toUpperCase();
|
||||||
|
}
|
||||||
|
console.log(this.word);
|
||||||
|
}
|
||||||
|
render(clear) {
|
||||||
|
h = g.getHeight();
|
||||||
|
bh = Math.floor(h/6);
|
||||||
|
bbh = Math.floor(0.85*bh);
|
||||||
|
w = g.getWidth();
|
||||||
|
bw = Math.floor(w/5);
|
||||||
|
bbw = Math.floor(0.85*bw);
|
||||||
|
if (clear) g.clear();
|
||||||
|
g.setFont("Vector", Math.floor(bbh*0.95)).setFontAlign(0,0);
|
||||||
|
g.setColor(g.theme.fg);
|
||||||
|
for (i=0; i<6; ++i) {
|
||||||
|
for (j=0; j<5; ++j) {
|
||||||
|
if (i<=this.nGuesses) {
|
||||||
|
g.setColor(this.guessColors[i][j]).fillRect(j*bw+(bw-bbw)/2, i*bh+(bh-bbh)/2, (j+1)*bw-(bw-bbw)/2, (i+1)*bh-(bh-bbh)/2);
|
||||||
|
g.setColor(g.theme.fg).drawString(this.guesses[i][j], 2+j*bw+bw/2, 2+i*bh+bh/2);
|
||||||
|
}
|
||||||
|
g.setColor(g.theme.fg).drawRect(j*bw+(bw-bbw)/2, i*bh+(bh-bbh)/2, (j+1)*bw-(bw-bbw)/2, (i+1)*bh-(bh-bbh)/2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
addGuess(w) {
|
||||||
|
if ((this.words.indexOf(w.toLowerCase())%5)!=0) {
|
||||||
|
E.showAlert(w+"\nis not a word", "Invalid word").then(function() {
|
||||||
|
layout = getKeyLayout("");
|
||||||
|
wordle.render(true);
|
||||||
|
});
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
this.guesses.push(w);
|
||||||
|
this.nGuesses++;
|
||||||
|
this.guessColors.push([]);
|
||||||
|
correct = 0;
|
||||||
|
var sol = this.word;
|
||||||
|
for (i=0; i<w.length; ++i) {
|
||||||
|
c = w[i];
|
||||||
|
col = g.theme.bg;
|
||||||
|
if (sol[i]==c) {
|
||||||
|
sol = sol.substr(0,i) + '?' + sol.substr(i+1);
|
||||||
|
col = "#0f0";
|
||||||
|
++correct;
|
||||||
|
}
|
||||||
|
else if (sol.includes(c)) col = "#ff0";
|
||||||
|
if (col!=g.theme.bg) this.keyColors[c] = this.keyColors[c] || col;
|
||||||
|
else this.keyColors[c] = "#00f";
|
||||||
|
this.guessColors[this.nGuesses].push(col);
|
||||||
|
}
|
||||||
|
if (correct==5) {
|
||||||
|
E.showAlert("The word is\n"+this.word, "You won in "+(this.nGuesses+1)+" guesses!").then(function(){load();});
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (this.nGuesses==5) {
|
||||||
|
E.showAlert("The word was\n"+this.word, "You lost!").then(function(){load();});
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wordle = new Wordle("rnd");
|
||||||
|
layout = getKeyLayout("");
|
||||||
|
wordle.render(true);
|
||||||
|
|
||||||
|
Bangle.on('swipe', function (dir) {
|
||||||
|
if (dir==1 || dir==-1) {
|
||||||
|
g.clear();
|
||||||
|
if (gameState==0) {
|
||||||
|
layout.render();
|
||||||
|
gameState = 1;
|
||||||
|
}
|
||||||
|
else if (gameState==1) {
|
||||||
|
wordle.render();
|
||||||
|
gameState = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
|
@ -0,0 +1,15 @@
|
||||||
|
{ "id": "bordle",
|
||||||
|
"name": "Bordle",
|
||||||
|
"shortName":"Bordle",
|
||||||
|
"icon": "app.png",
|
||||||
|
"version":"0.01",
|
||||||
|
"description": "Bangle version of a popular word search game",
|
||||||
|
"supports" : ["BANGLEJS2"],
|
||||||
|
"readme": "README.md",
|
||||||
|
"tags": "game, text",
|
||||||
|
"storage": [
|
||||||
|
{"name":"bordle.app.js","url":"app.js"},
|
||||||
|
{"name":"wordlencr.txt","url":"wordlencr.txt"}
|
||||||
|
{"name":"bordle.img","url":"app-icon.js","evaluate":true}
|
||||||
|
]
|
||||||
|
}
|
File diff suppressed because one or more lines are too long
Loading…
Reference in New Issue