Skip to content

Instantly share code, notes, and snippets.

@zydeco
Created February 12, 2023 12:18
Show Gist options
  • Save zydeco/a442496db92b0fd17cc9a258047e2e83 to your computer and use it in GitHub Desktop.
Save zydeco/a442496db92b0fd17cc9a258047e2e83 to your computer and use it in GitHub Desktop.
Cheers!
<!DOCTYPE html>
<html lang='en'>
<head>
<meta charset='utf-8'>
<meta http-equiv='x-ua-compatible' content='ie=edge'>
<meta name='viewport' content='width=device-width, initial-scale=1, shrink-to-fit=no'>
<title>Cheers!</title>
<style type='text/css'>
.ingredient {
background-size: contain;
background-repeat: no-repeat;
height: 92px;
}
.ingredient.r {
width: 84px;
content: url('');
}
.ingredient.g {
width: 68px;
content: url('');
}
.ingredient.b {
width: 76px;
content: url('');
}
.ingredient.p {
width: 84px;
content: url('');
}
.ingredient.y {
width: 52px;
content: url('');
}
div.main {
text-align: center;
margin-top: 2em;
}
button {
margin-top: 6em;
}
div.warning {
padding-top: 2em;
font-weight: bold;
font-size: 150%;
}
</style>
<script type='text/javascript'>
function shuffle(word) {
let array = word.split('');
let currentIndex = array.length,
randomIndex;
// While there remain elements to shuffle.
while (currentIndex != 0) {
// Pick a remaining element.
randomIndex = Math.floor(Math.random() * currentIndex);
currentIndex--;
// And swap it with the current element.
[array[currentIndex], array[randomIndex]] = [
array[randomIndex],
array[currentIndex]
];
}
return array.join('');
}
function sortw(word) {
return word.split('').sort().join('');
}
function makeRecipe() {
let drinks = shuffle('rygbp');
// first group: 2 of 2 kinds
let part1 = drinks.substring(0, 2) + drinks.substring(0, 2);
// second group: 2 of remaining 3 kinds
let part2 = drinks.substring(2) + drinks.substring(2);
// pick 3 from first group, 2 from second group
let recipe = shuffle(part1).substring(0, 3) + shuffle(part2).substring(0, 2);
return sortw(recipe);
}
function renderRecipe(recipe) {
let span = document.getElementById('recipe');
while (span.firstChild) {
span.removeChild(span.firstChild);
}
let ingredients = recipe.split('');
for (var i = 0; i < ingredients.length; i++) {
let ingredient = ingredients[i];
let img = document.createElement('img')
img.setAttribute('class', 'ingredient ' + ingredient);
span.appendChild(img);
}
if (!isLegalRecipe(recipe)) {
let warning = document.createElement('div');
warning.setAttribute('class', 'warning');
warning.appendChild(document.createTextNode('Invalid recipe!'));
span.appendChild(warning);
}
}
function isLegalRecipe(recipe) {
if (typeof(recipe) != 'string') {
return false;
}
let ingredients = recipe.split('');
// must have 5 ingredients
if (ingredients.length != 5) {
return false;
}
// every ingredient must be valid
let counter = { r: 0, g: 0, b: 0, y: 0, p: 0};
for (var i = 0; i < ingredients.length; i++) {
let ingredient = ingredients[i];
if ('rgbyp'.indexOf(ingredient) == -1) {
return false;
}
counter[ingredient] += 1;
}
// must have 2 of at least one ingredient
// can't have more than 2 of any ingredient
let counts = Object.values(counter);
let max = Math.max(...counts);
if (max != 2) {
return false;
}
return true;
}
function newRecipe() {
let recipe = makeRecipe();
renderRecipe(recipe);
history.pushState(null, null, '#' + recipe);
}
window.addEventListener('hashchange', (event) => {
let recipe = event.newURL.split('#', 2)[1] || 'rgbyp';
renderRecipe(recipe);
});
</script>
</head>
<body onLoad="renderRecipe(location.hash.substring(1) || 'rgbyp')">
<div class='main'>
<span id='recipe'>
<img class='ingredient r' />
<img class='ingredient g' />
<img class='ingredient b' />
<img class='ingredient y' />
<img class='ingredient p' />
</span>
<br/>
<button onClick='newRecipe()'>New Recipe</button>
</div>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment