manifold/web/pages/wordle/index.tsx

116 lines
3.5 KiB
TypeScript
Raw Normal View History

import { useState } from 'react'
2022-01-30 05:24:05 +00:00
import { Page } from '../../components/page'
import { Title } from '../../components/title'
// From https://github.com/lynn/hello-wordl
// `dictionary` has all accepted words; `targets` are common-ish words
import * as dictionary from './dictionary.json'
import * as targets from './targets.json'
export default function Wordle() {
const WORD_LENGTH = 5
// Find all words of length WORD_LENGTH
// const dict = dictionary.filter((word) => word.length === WORD_LENGTH)
const words = targets.filter((word) => word.length === WORD_LENGTH)
const [input, setInput] = useState('')
const pastGuesses: PastGuess[] = parsePastGuesses(input)
const placeholder = 'e.g.\nstage BBYBB\nlucre YYGYY'
2022-01-30 05:24:05 +00:00
const valids = validHardMode(words, pastGuesses)
return (
<Page>
<div className="text-gray-500">
<Title text="Wordle: Hard Mode possibilities" />
<p className="text-black">
Given a list of past Wordle guesses, show all possible valid next
guesses for Hard Mode
</p>
Actually for{' '}
<a href="https://hellowordl.net/">https://hellowordl.net/</a>
<p>Past guesses:</p>
<textarea
className="h-32 w-full p-2"
value={input}
onChange={(e) => setInput(e.target.value)}
placeholder={placeholder}
/>
2022-01-30 05:24:05 +00:00
<br />
<br />
<p className="text-black">
{Array(10)
.fill(null)
.map((_) => chooseRandom(valids))
.join(', ')}
</p>
<p>Total valid words: {valids.length}</p>
<p>All valid words:</p>
<pre>{JSON.stringify(valids, null, 2)}</pre>
</div>
</Page>
)
}
function chooseRandom(list: any[]) {
return list[Math.floor(Math.random() * list.length)]
}
type PastGuess = {
guess: string
// Result is a string like 'GYBBG'
// G = Green (match), Y = Yellow (somewhere), B = Black (nowhere)
result: string
}
function validHardMode(wordlist: string[], pastGuesses: PastGuess[]) {
let candidates = wordlist.slice()
const blacks = new Set()
for (const { guess, result } of pastGuesses) {
candidates = candidates.filter((word) => {
// e.g. guess = 'helot', result = 'GGGYB', word = 'hello => true
// e.g. guess = 'helot', result = 'GGGYB', word = 'helot => false
const guessYellows = []
const wordNonGreens: string[] = []
for (let i = 0; i < result.length; i++) {
switch (result[i]) {
case 'G':
if (word[i] !== guess[i]) return false
break
case 'Y':
if (word[i] === guess[i]) return false
guessYellows.push(guess[i])
wordNonGreens.push(word[i])
break
case 'B':
if (word[i] === guess[i]) return false
wordNonGreens.push(word[i])
blacks.add(guess[i])
break
}
if (blacks.has(word[i])) return false
}
// Return true if every letter in guessYellows is in wordNonGreens
return guessYellows.every((letter) => wordNonGreens.includes(letter))
})
}
return candidates
}
function parsePastGuesses(pastGuesses: string) {
return pastGuesses
.split('\n')
.map((pg) => {
const [guess, result] = pg.split(' ')
return { guess, result }
})
.filter(({ guess, result }) => guess && result)
2022-01-30 05:24:05 +00:00
}
// Bad target words: witan, sedum
// TODO: Maybe just use canonical Wordle dictionaries
// DEBUG: { guess: 'lucre', result: 'YYGYY' }
// Should produce: ulcer