Setup basic word mutating system
Next up: prettification!
This commit is contained in:
commit
cb9ac2640f
6 changed files with 332 additions and 0 deletions
160
.gitignore
vendored
Normal file
160
.gitignore
vendored
Normal file
|
@ -0,0 +1,160 @@
|
||||||
|
|
||||||
|
# Created by https://www.toptal.com/developers/gitignore/api/git,node
|
||||||
|
# Edit at https://www.toptal.com/developers/gitignore?templates=git,node
|
||||||
|
|
||||||
|
### Git ###
|
||||||
|
# Created by git for backups. To disable backups in Git:
|
||||||
|
# $ git config --global mergetool.keepBackup false
|
||||||
|
*.orig
|
||||||
|
|
||||||
|
# Created by git when using merge tools for conflicts
|
||||||
|
*.BACKUP.*
|
||||||
|
*.BASE.*
|
||||||
|
*.LOCAL.*
|
||||||
|
*.REMOTE.*
|
||||||
|
*_BACKUP_*.txt
|
||||||
|
*_BASE_*.txt
|
||||||
|
*_LOCAL_*.txt
|
||||||
|
*_REMOTE_*.txt
|
||||||
|
|
||||||
|
### Node ###
|
||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
lerna-debug.log*
|
||||||
|
.pnpm-debug.log*
|
||||||
|
|
||||||
|
# Diagnostic reports (https://nodejs.org/api/report.html)
|
||||||
|
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
|
||||||
|
|
||||||
|
# Runtime data
|
||||||
|
pids
|
||||||
|
*.pid
|
||||||
|
*.seed
|
||||||
|
*.pid.lock
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
*.lcov
|
||||||
|
|
||||||
|
# nyc test coverage
|
||||||
|
.nyc_output
|
||||||
|
|
||||||
|
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||||
|
.grunt
|
||||||
|
|
||||||
|
# Bower dependency directory (https://bower.io/)
|
||||||
|
bower_components
|
||||||
|
|
||||||
|
# node-waf configuration
|
||||||
|
.lock-wscript
|
||||||
|
|
||||||
|
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||||
|
build/Release
|
||||||
|
|
||||||
|
# Dependency directories
|
||||||
|
node_modules/
|
||||||
|
jspm_packages/
|
||||||
|
|
||||||
|
# Snowpack dependency directory (https://snowpack.dev/)
|
||||||
|
web_modules/
|
||||||
|
|
||||||
|
# TypeScript cache
|
||||||
|
*.tsbuildinfo
|
||||||
|
|
||||||
|
# Optional npm cache directory
|
||||||
|
.npm
|
||||||
|
|
||||||
|
# Optional eslint cache
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Optional stylelint cache
|
||||||
|
.stylelintcache
|
||||||
|
|
||||||
|
# Microbundle cache
|
||||||
|
.rpt2_cache/
|
||||||
|
.rts2_cache_cjs/
|
||||||
|
.rts2_cache_es/
|
||||||
|
.rts2_cache_umd/
|
||||||
|
|
||||||
|
# Optional REPL history
|
||||||
|
.node_repl_history
|
||||||
|
|
||||||
|
# Output of 'npm pack'
|
||||||
|
*.tgz
|
||||||
|
|
||||||
|
# Yarn Integrity file
|
||||||
|
.yarn-integrity
|
||||||
|
|
||||||
|
# dotenv environment variable files
|
||||||
|
.env
|
||||||
|
.env.development.local
|
||||||
|
.env.test.local
|
||||||
|
.env.production.local
|
||||||
|
.env.local
|
||||||
|
|
||||||
|
# parcel-bundler cache (https://parceljs.org/)
|
||||||
|
.cache
|
||||||
|
.parcel-cache
|
||||||
|
|
||||||
|
# Next.js build output
|
||||||
|
.next
|
||||||
|
out
|
||||||
|
|
||||||
|
# Nuxt.js build / generate output
|
||||||
|
.nuxt
|
||||||
|
dist
|
||||||
|
|
||||||
|
# Gatsby files
|
||||||
|
.cache/
|
||||||
|
# Comment in the public line in if your project uses Gatsby and not Next.js
|
||||||
|
# https://nextjs.org/blog/next-9-1#public-directory-support
|
||||||
|
# public
|
||||||
|
|
||||||
|
# vuepress build output
|
||||||
|
.vuepress/dist
|
||||||
|
|
||||||
|
# vuepress v2.x temp and cache directory
|
||||||
|
.temp
|
||||||
|
|
||||||
|
# Docusaurus cache and generated files
|
||||||
|
.docusaurus
|
||||||
|
|
||||||
|
# Serverless directories
|
||||||
|
.serverless/
|
||||||
|
|
||||||
|
# FuseBox cache
|
||||||
|
.fusebox/
|
||||||
|
|
||||||
|
# DynamoDB Local files
|
||||||
|
.dynamodb/
|
||||||
|
|
||||||
|
# TernJS port file
|
||||||
|
.tern-port
|
||||||
|
|
||||||
|
# Stores VSCode versions used for testing VSCode extensions
|
||||||
|
.vscode-test
|
||||||
|
|
||||||
|
# yarn v2
|
||||||
|
.yarn/cache
|
||||||
|
.yarn/unplugged
|
||||||
|
.yarn/build-state.yml
|
||||||
|
.yarn/install-state.gz
|
||||||
|
.pnp.*
|
||||||
|
|
||||||
|
### Node Patch ###
|
||||||
|
# Serverless Webpack directories
|
||||||
|
.webpack/
|
||||||
|
|
||||||
|
# Optional stylelint cache
|
||||||
|
|
||||||
|
# SvelteKit build / generate output
|
||||||
|
.svelte-kit
|
||||||
|
|
||||||
|
# End of https://www.toptal.com/developers/gitignore/api/git,node
|
32
css/theme.css
Normal file
32
css/theme.css
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
:root {
|
||||||
|
--bg: #bff4fb;
|
||||||
|
--accent: #333333;
|
||||||
|
--accent-dark: #777777;
|
||||||
|
/* --bg: #038de8;
|
||||||
|
--accent: #e87e04;
|
||||||
|
--accent-dark: #d55400; */
|
||||||
|
}
|
||||||
|
|
||||||
|
html, body { font-size: 100%; }
|
||||||
|
body, input[type=text] {
|
||||||
|
font-family: "Ubuntu", sans-serif;
|
||||||
|
text-align: center;
|
||||||
|
font-size: 150%;
|
||||||
|
|
||||||
|
background: var(--bg);
|
||||||
|
color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type=text] {
|
||||||
|
border: 0;
|
||||||
|
border-bottom: 0.2em solid var(--accent-dark);
|
||||||
|
}
|
||||||
|
|
||||||
|
.mutation-step {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-evenly;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mutation-option {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
24
index.html
Normal file
24
index.html
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset='utf-8' />
|
||||||
|
<title>mutate-a-word!</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>mutate-a-word!</h1>
|
||||||
|
<section>
|
||||||
|
<h2>Enter a starting word</h2>
|
||||||
|
<p>
|
||||||
|
<input type="text" id="starting-word" value="" placeholder="e.g. rockets" />
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section id="mutations">
|
||||||
|
<em>Mutations will appear here</em>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!---------------->
|
||||||
|
<link rel="stylesheet" href="./css/theme.css" />
|
||||||
|
<script src="./js/index.mjs" charset="utf-8" type="module"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
68
js/index.mjs
Normal file
68
js/index.mjs
Normal file
|
@ -0,0 +1,68 @@
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
import mutate from './mutate.mjs';
|
||||||
|
|
||||||
|
let display_options = 3;
|
||||||
|
|
||||||
|
|
||||||
|
window.addEventListener("load", (_event) => {
|
||||||
|
const el_starting_word = document.querySelector("#starting-word");
|
||||||
|
|
||||||
|
if(el_starting_word.value.length > 0)
|
||||||
|
handle_starting_keyup();
|
||||||
|
|
||||||
|
el_starting_word.addEventListener("keyup", handle_starting_keyup);
|
||||||
|
});
|
||||||
|
|
||||||
|
function handle_starting_keyup(_event) {
|
||||||
|
const el_starting_word = document.querySelector("#starting-word");
|
||||||
|
const el_mutations = document.querySelector("#mutations");
|
||||||
|
|
||||||
|
el_mutations.replaceChildren(); // Empty node of all children
|
||||||
|
|
||||||
|
mutation_step(el_starting_word.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_mutation_word_keyup(event) {
|
||||||
|
// 1: Find DOM elements
|
||||||
|
const el_step = event.target.closest(".mutation-step");
|
||||||
|
const el_word = event.target.closest(".mutation-option");
|
||||||
|
|
||||||
|
// 2: Update UI - Add classes, Remove all subsequent mutation steps
|
||||||
|
el_word.classList.add("selected");
|
||||||
|
while(el_step.nextElementSibling !== null)
|
||||||
|
el_step.parentNode.removeChild(el_step.nextElementSibling);
|
||||||
|
|
||||||
|
// 3: Do a new mutation step
|
||||||
|
mutation_step(el_word.dataset.word);
|
||||||
|
}
|
||||||
|
|
||||||
|
function mutation_step(word) {
|
||||||
|
// 0: Validate input
|
||||||
|
if(typeof word !== "string")
|
||||||
|
throw new Error(`Error: Expected argument 1 (word) to be of type string, but got value of type ${typeof word}.`);
|
||||||
|
|
||||||
|
// 1: Find DOM elements
|
||||||
|
const el_mutations = document.querySelector("#mutations");
|
||||||
|
|
||||||
|
// 2: Generate mutations
|
||||||
|
// TODO: Eliminate duplicates
|
||||||
|
const words_new = Array(display_options).fill(null)
|
||||||
|
.map(() => mutate(word));
|
||||||
|
|
||||||
|
// 3: Update DOM
|
||||||
|
let el_step = document.createElement("div");
|
||||||
|
el_step.classList.add("mutation-step");
|
||||||
|
for(let word_item of words_new) {
|
||||||
|
let el_word = document.createElement("span");
|
||||||
|
el_word.classList.add("mutation-option");
|
||||||
|
el_word.appendChild(document.createTextNode(word_item));
|
||||||
|
el_word.dataset.word = word_item;
|
||||||
|
el_step.appendChild(el_word);
|
||||||
|
el_word.addEventListener("click", handle_mutation_word_keyup);
|
||||||
|
el_word.addEventListener("touchend", handle_mutation_word_keyup);
|
||||||
|
el_word.addEventListener("mouseup", handle_mutation_word_keyup);
|
||||||
|
}
|
||||||
|
|
||||||
|
el_mutations.appendChild(el_step);
|
||||||
|
}
|
33
js/mutate.mjs
Normal file
33
js/mutate.mjs
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const vowels = "aeiou".split("");
|
||||||
|
const consonants = "bcdfghjklmnpqrstvwxyz".split("");
|
||||||
|
|
||||||
|
function random_item(arr) {
|
||||||
|
return arr[Math.floor(Math.random() * arr.length)];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function(word) {
|
||||||
|
console.log(`***** MUTATE *****`)
|
||||||
|
const chars = word.toLowerCase().split("");
|
||||||
|
const targetpos = Math.floor(Math.random() * word.length);
|
||||||
|
const targetchar = chars[targetpos];
|
||||||
|
console.log("TARGET", targetchar, "POS", targetpos);
|
||||||
|
let pool = consonants.concat(vowels);
|
||||||
|
if(vowels.includes(targetchar))
|
||||||
|
pool = vowels;
|
||||||
|
if(consonants.includes(targetchar))
|
||||||
|
pool = consonants;
|
||||||
|
|
||||||
|
pool = [...pool]; // Shallow copy to avoid splice mutating the original array
|
||||||
|
console.log("POOL BEFORE", pool);
|
||||||
|
if(pool.includes(targetchar)) {
|
||||||
|
console.log("REMOVING TARGET");
|
||||||
|
pool.splice(pool.indexOf(targetchar), 1);
|
||||||
|
}
|
||||||
|
console.log("POOL AFTER", pool);
|
||||||
|
|
||||||
|
let newchar = random_item(pool);
|
||||||
|
chars.splice(targetpos, 1, newchar);
|
||||||
|
return chars.join("");
|
||||||
|
}
|
15
package.json
Normal file
15
package.json
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
{
|
||||||
|
"name": "mutate-a-word",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Mutate a word to find a cool name!",
|
||||||
|
"main": "js/index.mjs",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://git.starbeamrainbowlabs.com/sbrl/mutate-a-word.git"
|
||||||
|
},
|
||||||
|
"author": "Starbeamrainbowlabs",
|
||||||
|
"license": "MPL-2.0"
|
||||||
|
}
|
Loading…
Reference in a new issue