puzzle-lib 2.0.0-alpha.2 → 2.0.0-beta.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +62 -9
- package/build/src/{Braille → braille}/braille.d.ts +4 -4
- package/build/src/braille/braille.js +121 -0
- package/build/src/braille/braille.js.map +1 -0
- package/build/src/{Braille/BrailleDot.js → braille/dot.js} +3 -6
- package/build/src/braille/dot.js.map +1 -0
- package/build/src/{Braille/BrailleEncoding.js → braille/encoding.js} +4 -7
- package/build/src/braille/encoding.js.map +1 -0
- package/build/src/braille/index.d.ts +3 -0
- package/build/src/braille/index.js +4 -0
- package/build/src/braille/index.js.map +1 -0
- package/build/src/{Cipher → cipher}/cipher.js +10 -20
- package/build/src/cipher/cipher.js.map +1 -0
- package/build/src/cipher/index.d.ts +1 -0
- package/build/src/cipher/index.js +2 -0
- package/build/src/cipher/index.js.map +1 -0
- package/build/src/{Common/CharacterImage.d.ts → common/character-image.d.ts} +1 -1
- package/build/src/common/character-image.js +9 -0
- package/build/src/common/character-image.js.map +1 -0
- package/build/src/{Common/EncodingCategory.js → common/encoding-category.js} +3 -6
- package/build/src/common/encoding-category.js.map +1 -0
- package/build/src/{Common/EncodingEntry.d.ts → common/encoding-entry.d.ts} +1 -1
- package/build/src/{Common/EncodingEntry.js → common/encoding-entry.js} +2 -6
- package/build/src/common/encoding-entry.js.map +1 -0
- package/build/src/{Common/EncodingLookupResult.d.ts → common/encoding-lookup-result.d.ts} +1 -1
- package/build/src/common/encoding-lookup-result.js +8 -0
- package/build/src/common/encoding-lookup-result.js.map +1 -0
- package/build/src/common/index.d.ts +5 -0
- package/build/src/common/index.js +6 -0
- package/build/src/common/index.js.map +1 -0
- package/build/src/{Common/InlineSvg.js → common/inline-svg.js} +2 -6
- package/build/src/common/inline-svg.js.map +1 -0
- package/build/src/{Conversion/characterConversion.d.ts → conversion/character-conversion.d.ts} +1 -1
- package/build/src/{Conversion/characterConversion.js → conversion/character-conversion.js} +7 -13
- package/build/src/conversion/character-conversion.js.map +1 -0
- package/build/src/{Conversion/CharacterEncoding.d.ts → conversion/character-encoding.d.ts} +2 -1
- package/build/src/{Conversion/CharacterEncoding.js → conversion/character-encoding.js} +4 -6
- package/build/src/conversion/character-encoding.js.map +1 -0
- package/build/src/{Conversion/CharacterTableEntry.js → conversion/character-table-entry.js} +2 -6
- package/build/src/conversion/character-table-entry.js.map +1 -0
- package/build/src/{Conversion → conversion}/conversion.d.ts +3 -1
- package/build/src/conversion/conversion.js +183 -0
- package/build/src/conversion/conversion.js.map +1 -0
- package/build/src/conversion/index.d.ts +5 -0
- package/build/src/conversion/index.js +6 -0
- package/build/src/conversion/index.js.map +1 -0
- package/build/src/{Conversion/significantFigures.js → conversion/significant-figures.js} +4 -9
- package/build/src/conversion/significant-figures.js.map +1 -0
- package/build/src/{Morse/MorseEncoding.js → morse/encoding.js} +3 -6
- package/build/src/morse/encoding.js.map +1 -0
- package/build/src/morse/index.d.ts +2 -0
- package/build/src/morse/index.js +3 -0
- package/build/src/morse/index.js.map +1 -0
- package/build/src/{Morse → morse}/morse.d.ts +5 -5
- package/build/src/morse/morse.js +222 -0
- package/build/src/morse/morse.js.map +1 -0
- package/build/src/nato/character.js +9 -0
- package/build/src/nato/character.js.map +1 -0
- package/build/src/nato/index.d.ts +2 -0
- package/build/src/nato/index.js +3 -0
- package/build/src/nato/index.js.map +1 -0
- package/build/src/{Nato → nato}/nato.d.ts +1 -1
- package/build/src/nato/nato.js +33 -0
- package/build/src/nato/nato.js.map +1 -0
- package/build/src/naval-flags/index.d.ts +1 -0
- package/build/src/naval-flags/index.js +2 -0
- package/build/src/naval-flags/index.js.map +1 -0
- package/build/src/{NavalFlags/navalFlags.d.ts → naval-flags/naval-flags.d.ts} +1 -1
- package/build/src/{NavalFlags/navalFlags.js → naval-flags/naval-flags.js} +32 -36
- package/build/src/naval-flags/naval-flags.js.map +1 -0
- package/build/src/ngrams/bigrams.d.ts +2 -0
- package/build/src/ngrams/bigrams.js +684 -0
- package/build/src/ngrams/bigrams.js.map +1 -0
- package/build/src/ngrams/index.d.ts +1 -0
- package/build/src/ngrams/index.js +2 -0
- package/build/src/ngrams/index.js.map +1 -0
- package/build/src/ngrams/ngrams.d.ts +13 -0
- package/build/src/ngrams/ngrams.js +71 -0
- package/build/src/ngrams/ngrams.js.map +1 -0
- package/build/src/ngrams/trigrams.d.ts +2 -0
- package/build/src/ngrams/trigrams.js +17583 -0
- package/build/src/ngrams/trigrams.js.map +1 -0
- package/build/src/ngrams/unigrams.d.ts +2 -0
- package/build/src/ngrams/unigrams.js +34 -0
- package/build/src/ngrams/unigrams.js.map +1 -0
- package/build/src/phone/index.d.ts +2 -0
- package/build/src/phone/index.js +2 -0
- package/build/src/phone/index.js.map +1 -0
- package/build/src/phone/phone-mapping.d.ts +10 -0
- package/build/src/phone/phone-mapping.js +27 -0
- package/build/src/phone/phone-mapping.js.map +1 -0
- package/build/src/phone/phone.d.ts +27 -0
- package/build/src/phone/phone.js +62 -0
- package/build/src/phone/phone.js.map +1 -0
- package/build/src/pigpen/encoding.d.ts +29 -0
- package/build/src/pigpen/encoding.js +36 -0
- package/build/src/pigpen/encoding.js.map +1 -0
- package/build/src/pigpen/index.d.ts +3 -0
- package/build/src/pigpen/index.js +4 -0
- package/build/src/pigpen/index.js.map +1 -0
- package/build/src/pigpen/pigpen.d.ts +35 -0
- package/build/src/pigpen/pigpen.js +137 -0
- package/build/src/pigpen/pigpen.js.map +1 -0
- package/build/src/pigpen/segment.d.ts +12 -0
- package/build/src/pigpen/segment.js +14 -0
- package/build/src/pigpen/segment.js.map +1 -0
- package/build/src/resistor/index.d.ts +2 -0
- package/build/src/resistor/index.js +2 -0
- package/build/src/resistor/index.js.map +1 -0
- package/build/src/{Resistor → resistor}/resistor.js +7 -14
- package/build/src/resistor/resistor.js.map +1 -0
- package/build/src/{Semaphore/SemaphoreDirection.js → semaphore/direction.js} +3 -6
- package/build/src/semaphore/direction.js.map +1 -0
- package/build/src/{Semaphore/SemaphoreEncoding.js → semaphore/encoding.js} +4 -7
- package/build/src/semaphore/encoding.js.map +1 -0
- package/build/src/semaphore/index.d.ts +3 -0
- package/build/src/semaphore/index.js +4 -0
- package/build/src/semaphore/index.js.map +1 -0
- package/build/src/{Semaphore → semaphore}/semaphore.d.ts +4 -4
- package/build/src/semaphore/semaphore.js +207 -0
- package/build/src/semaphore/semaphore.js.map +1 -0
- package/build/src/{WordSearch/WordSearchDirection.js → word-search/direction.js} +3 -6
- package/build/src/word-search/direction.js.map +1 -0
- package/build/src/word-search/index.d.ts +5 -0
- package/build/src/word-search/index.js +4 -0
- package/build/src/word-search/index.js.map +1 -0
- package/build/src/word-search/point.d.ts +4 -0
- package/build/src/word-search/point.js +2 -0
- package/build/src/word-search/point.js.map +1 -0
- package/build/src/word-search/result.d.ts +6 -0
- package/build/src/word-search/result.js +9 -0
- package/build/src/word-search/result.js.map +1 -0
- package/build/src/word-search/word-search.d.ts +10 -0
- package/build/src/{WordSearch/wordSearch.js → word-search/word-search.js} +28 -55
- package/build/src/word-search/word-search.js.map +1 -0
- package/package.json +57 -3
- package/src/{Braille → braille}/braille.ts +5 -5
- package/src/{Braille/BrailleEncoding.ts → braille/encoding.ts} +1 -1
- package/src/braille/index.ts +8 -0
- package/src/{Cipher → cipher}/cipher.ts +1 -1
- package/src/cipher/index.ts +10 -0
- package/src/{Common/CharacterImage.ts → common/character-image.ts} +1 -1
- package/src/{Common/EncodingEntry.ts → common/encoding-entry.ts} +1 -1
- package/src/{Common/EncodingLookupResult.ts → common/encoding-lookup-result.ts} +1 -1
- package/src/common/index.ts +5 -0
- package/src/{Conversion/characterConversion.ts → conversion/character-conversion.ts} +1 -1
- package/src/{Conversion/CharacterEncoding.ts → conversion/character-encoding.ts} +1 -0
- package/src/{Conversion → conversion}/conversion.ts +77 -2
- package/src/conversion/index.ts +15 -0
- package/src/morse/index.ts +14 -0
- package/src/{Morse → morse}/morse.ts +6 -6
- package/src/nato/index.ts +2 -0
- package/src/{Nato → nato}/nato.ts +1 -1
- package/src/naval-flags/index.ts +1 -0
- package/src/{NavalFlags/navalFlags.ts → naval-flags/naval-flags.ts} +2 -2
- package/src/ngrams/bigrams.ts +684 -0
- package/src/ngrams/index.ts +1 -0
- package/src/ngrams/ngrams.ts +73 -0
- package/src/ngrams/trigrams.ts +17584 -0
- package/src/ngrams/unigrams.ts +34 -0
- package/src/phone/index.ts +2 -0
- package/src/phone/phone-mapping.ts +27 -0
- package/src/phone/phone.ts +77 -0
- package/src/pigpen/encoding.ts +66 -0
- package/src/pigpen/index.ts +11 -0
- package/src/pigpen/pigpen.ts +168 -0
- package/src/pigpen/segment.ts +12 -0
- package/src/resistor/index.ts +9 -0
- package/src/{Semaphore/SemaphoreEncoding.ts → semaphore/encoding.ts} +1 -1
- package/src/semaphore/index.ts +13 -0
- package/src/{Semaphore → semaphore}/semaphore.ts +5 -5
- package/src/trie-prefix-tree.d.ts +1 -3
- package/src/word-search/index.ts +5 -0
- package/src/word-search/point.ts +4 -0
- package/src/word-search/result.ts +10 -0
- package/src/{WordSearch/wordSearch.ts → word-search/word-search.ts} +44 -70
- package/build/src/Braille/BrailleDot.js.map +0 -1
- package/build/src/Braille/BrailleEncoding.js.map +0 -1
- package/build/src/Braille/braille.js +0 -127
- package/build/src/Braille/braille.js.map +0 -1
- package/build/src/Cipher/cipher.js.map +0 -1
- package/build/src/Common/CharacterImage.js +0 -13
- package/build/src/Common/CharacterImage.js.map +0 -1
- package/build/src/Common/EncodingCategory.js.map +0 -1
- package/build/src/Common/EncodingEntry.js.map +0 -1
- package/build/src/Common/EncodingLookupResult.js +0 -12
- package/build/src/Common/EncodingLookupResult.js.map +0 -1
- package/build/src/Common/InlineSvg.js.map +0 -1
- package/build/src/Conversion/CharacterEncoding.js.map +0 -1
- package/build/src/Conversion/CharacterTableEntry.js.map +0 -1
- package/build/src/Conversion/characterConversion.js.map +0 -1
- package/build/src/Conversion/conversion.js +0 -130
- package/build/src/Conversion/conversion.js.map +0 -1
- package/build/src/Conversion/significantFigures.js.map +0 -1
- package/build/src/Morse/MorseEncoding.js.map +0 -1
- package/build/src/Morse/morse.js +0 -232
- package/build/src/Morse/morse.js.map +0 -1
- package/build/src/Nato/NatoCharacter.js +0 -13
- package/build/src/Nato/NatoCharacter.js.map +0 -1
- package/build/src/Nato/nato.js +0 -37
- package/build/src/Nato/nato.js.map +0 -1
- package/build/src/NavalFlags/navalFlags.js.map +0 -1
- package/build/src/Resistor/resistor.js.map +0 -1
- package/build/src/Semaphore/SemaphoreDirection.js.map +0 -1
- package/build/src/Semaphore/SemaphoreEncoding.js.map +0 -1
- package/build/src/Semaphore/semaphore.js +0 -218
- package/build/src/Semaphore/semaphore.js.map +0 -1
- package/build/src/WordSearch/Point.d.ts +0 -4
- package/build/src/WordSearch/Point.js +0 -3
- package/build/src/WordSearch/Point.js.map +0 -1
- package/build/src/WordSearch/Result.d.ts +0 -6
- package/build/src/WordSearch/Result.js +0 -13
- package/build/src/WordSearch/Result.js.map +0 -1
- package/build/src/WordSearch/WordSearchDirection.js.map +0 -1
- package/build/src/WordSearch/wordSearch.d.ts +0 -16
- package/build/src/WordSearch/wordSearch.js.map +0 -1
- package/build/src/index.d.ts +0 -29
- package/build/src/index.js +0 -102
- package/build/src/index.js.map +0 -1
- package/src/WordSearch/Point.ts +0 -4
- package/src/WordSearch/Result.ts +0 -10
- package/src/index.ts +0 -90
- /package/build/src/{Braille/BrailleDot.d.ts → braille/dot.d.ts} +0 -0
- /package/build/src/{Braille/BrailleEncoding.d.ts → braille/encoding.d.ts} +0 -0
- /package/build/src/{Cipher → cipher}/cipher.d.ts +0 -0
- /package/build/src/{Common/EncodingCategory.d.ts → common/encoding-category.d.ts} +0 -0
- /package/build/src/{Common/InlineSvg.d.ts → common/inline-svg.d.ts} +0 -0
- /package/build/src/{Conversion/CharacterTableEntry.d.ts → conversion/character-table-entry.d.ts} +0 -0
- /package/build/src/{Conversion/significantFigures.d.ts → conversion/significant-figures.d.ts} +0 -0
- /package/build/src/{Morse/MorseEncoding.d.ts → morse/encoding.d.ts} +0 -0
- /package/build/src/{Nato/NatoCharacter.d.ts → nato/character.d.ts} +0 -0
- /package/build/src/{Resistor → resistor}/resistor.d.ts +0 -0
- /package/build/src/{Semaphore/SemaphoreDirection.d.ts → semaphore/direction.d.ts} +0 -0
- /package/build/src/{Semaphore/SemaphoreEncoding.d.ts → semaphore/encoding.d.ts} +0 -0
- /package/build/src/{WordSearch/WordSearchDirection.d.ts → word-search/direction.d.ts} +0 -0
- /package/src/{Braille/BrailleDot.ts → braille/dot.ts} +0 -0
- /package/src/{Common/EncodingCategory.ts → common/encoding-category.ts} +0 -0
- /package/src/{Common/InlineSvg.ts → common/inline-svg.ts} +0 -0
- /package/src/{Conversion/CharacterTableEntry.ts → conversion/character-table-entry.ts} +0 -0
- /package/src/{Conversion/significantFigures.ts → conversion/significant-figures.ts} +0 -0
- /package/src/{Morse/MorseEncoding.ts → morse/encoding.ts} +0 -0
- /package/src/{Nato/NatoCharacter.ts → nato/character.ts} +0 -0
- /package/src/{NavalFlags → naval-flags}/LICENSE +0 -0
- /package/src/{Resistor → resistor}/resistor.ts +0 -0
- /package/src/{Semaphore/SemaphoreDirection.ts → semaphore/direction.ts} +0 -0
- /package/src/{WordSearch/WordSearchDirection.ts → word-search/direction.ts} +0 -0
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
// Log-probabilities of English letter unigrams.
|
|
2
|
+
// Derived from bigram frequencies (sum of P(xy) over all y for each x).
|
|
3
|
+
// Original bigram source: Peter Norvig, Google Web Trillion Word Corpus (public domain)
|
|
4
|
+
// http://norvig.com/ngrams/count_2l.txt
|
|
5
|
+
export const UNIGRAM_LOG_PROB: Record<string, number> = {
|
|
6
|
+
e: -2.186224,
|
|
7
|
+
a: -2.448764,
|
|
8
|
+
t: -2.47288,
|
|
9
|
+
o: -2.501531,
|
|
10
|
+
i: -2.516878,
|
|
11
|
+
n: -2.681277,
|
|
12
|
+
r: -2.700564,
|
|
13
|
+
s: -2.800337,
|
|
14
|
+
l: -3.133597,
|
|
15
|
+
c: -3.206786,
|
|
16
|
+
h: -3.266123,
|
|
17
|
+
d: -3.428238,
|
|
18
|
+
u: -3.464367,
|
|
19
|
+
m: -3.543925,
|
|
20
|
+
p: -3.613343,
|
|
21
|
+
f: -3.87574,
|
|
22
|
+
g: -3.926056,
|
|
23
|
+
b: -4.025837,
|
|
24
|
+
w: -4.158706,
|
|
25
|
+
y: -4.284818,
|
|
26
|
+
v: -4.386163,
|
|
27
|
+
k: -4.895394,
|
|
28
|
+
x: -5.875251,
|
|
29
|
+
j: -5.90304,
|
|
30
|
+
q: -6.558017,
|
|
31
|
+
z: -6.560645,
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
export const MIN_UNIGRAM_LOG_PROB = -11.560645;
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Standard telephone keypad mapping (ITU E.161 / T9).
|
|
3
|
+
* Maps digit characters to their corresponding letters.
|
|
4
|
+
*/
|
|
5
|
+
export const PHONE_MAPPING: Partial<Record<string, readonly string[]>> = {
|
|
6
|
+
'2': ['a', 'b', 'c'],
|
|
7
|
+
'3': ['d', 'e', 'f'],
|
|
8
|
+
'4': ['g', 'h', 'i'],
|
|
9
|
+
'5': ['j', 'k', 'l'],
|
|
10
|
+
'6': ['m', 'n', 'o'],
|
|
11
|
+
'7': ['p', 'q', 'r', 's'],
|
|
12
|
+
'8': ['t', 'u', 'v'],
|
|
13
|
+
'9': ['w', 'x', 'y', 'z'],
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Reverse mapping: letter to digit.
|
|
18
|
+
* Built automatically from PHONE_MAPPING.
|
|
19
|
+
*/
|
|
20
|
+
export const LETTER_TO_DIGIT: Record<string, string> = {};
|
|
21
|
+
for (const [digit, letters] of Object.entries(PHONE_MAPPING)) {
|
|
22
|
+
if (letters) {
|
|
23
|
+
for (const letter of letters) {
|
|
24
|
+
LETTER_TO_DIGIT[letter] = digit;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import {scoreNextLetter} from '../ngrams/ngrams.js';
|
|
2
|
+
import {LETTER_TO_DIGIT, PHONE_MAPPING} from './phone-mapping.js';
|
|
3
|
+
|
|
4
|
+
export interface PhoneResult {
|
|
5
|
+
/** The candidate text string. */
|
|
6
|
+
text: string;
|
|
7
|
+
/** Log-probability score (higher = more likely). */
|
|
8
|
+
score: number;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Returns the possible letters for a single digit.
|
|
13
|
+
* Returns an empty array for digits without letter mappings (0, 1).
|
|
14
|
+
*/
|
|
15
|
+
export function phoneToLetters(digit: string): readonly string[] {
|
|
16
|
+
return PHONE_MAPPING[digit] ?? [];
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Converts a string of letters to their corresponding phone digits.
|
|
21
|
+
* Non-letter characters are passed through unchanged.
|
|
22
|
+
*/
|
|
23
|
+
export function lettersToPhone(text: string): string {
|
|
24
|
+
return text
|
|
25
|
+
.split('')
|
|
26
|
+
.map(ch => LETTER_TO_DIGIT[ch.toLowerCase()] ?? ch)
|
|
27
|
+
.join('');
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Converts a string of phone digits (2-9) to ranked text candidates
|
|
32
|
+
* using beam search with n-gram scoring (unigram + bigram + trigram).
|
|
33
|
+
* Starts with unigram scoring for the first letter, bigram for the
|
|
34
|
+
* second, and trigrams for all subsequent letters.
|
|
35
|
+
*
|
|
36
|
+
* @param digits - String of digit characters (e.g., "43556")
|
|
37
|
+
* @param maxResults - Maximum number of results to return (default: 20, clamped to [1, 1000])
|
|
38
|
+
* @returns Array of PhoneResult sorted by score (highest first)
|
|
39
|
+
*/
|
|
40
|
+
export function phoneToText(digits: string, maxResults = 20): PhoneResult[] {
|
|
41
|
+
const clampedMax = Math.max(1, Math.min(Math.trunc(maxResults) || 1, 1000));
|
|
42
|
+
const beamWidth = Math.max(100, clampedMax);
|
|
43
|
+
// Filter to valid digits with letter mappings
|
|
44
|
+
const validDigits = digits.split('').filter(d => PHONE_MAPPING[d]);
|
|
45
|
+
|
|
46
|
+
if (validDigits.length === 0) {
|
|
47
|
+
return [];
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Start with letters from the first digit, scored by unigram frequency
|
|
51
|
+
let candidates: PhoneResult[] = phoneToLetters(validDigits[0]).map(ch => ({
|
|
52
|
+
text: ch,
|
|
53
|
+
score: scoreNextLetter('', ch),
|
|
54
|
+
}));
|
|
55
|
+
|
|
56
|
+
// Expand one digit at a time with beam search
|
|
57
|
+
for (let i = 1; i < validDigits.length; i++) {
|
|
58
|
+
const letters = phoneToLetters(validDigits[i]);
|
|
59
|
+
const next: PhoneResult[] = [];
|
|
60
|
+
|
|
61
|
+
for (const candidate of candidates) {
|
|
62
|
+
for (const letter of letters) {
|
|
63
|
+
const text = candidate.text + letter;
|
|
64
|
+
const score = candidate.score + scoreNextLetter(candidate.text, letter);
|
|
65
|
+
next.push({text, score});
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Keep only top beamWidth candidates
|
|
70
|
+
next.sort((a, b) => b.score - a.score);
|
|
71
|
+
candidates = next.slice(0, beamWidth);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Final sort and trim
|
|
75
|
+
candidates.sort((a, b) => b.score - a.score);
|
|
76
|
+
return candidates.slice(0, clampedMax);
|
|
77
|
+
}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import {PigpenSegment} from './segment.js';
|
|
2
|
+
|
|
3
|
+
export enum PigpenEncoding {
|
|
4
|
+
None = 0,
|
|
5
|
+
|
|
6
|
+
// Grid without dot (A-I)
|
|
7
|
+
LetterA = PigpenSegment.East | PigpenSegment.South,
|
|
8
|
+
LetterB = PigpenSegment.West | PigpenSegment.East | PigpenSegment.South,
|
|
9
|
+
LetterC = PigpenSegment.West | PigpenSegment.South,
|
|
10
|
+
LetterD = PigpenSegment.North | PigpenSegment.East | PigpenSegment.South,
|
|
11
|
+
LetterE = PigpenSegment.North |
|
|
12
|
+
PigpenSegment.East |
|
|
13
|
+
PigpenSegment.South |
|
|
14
|
+
PigpenSegment.West,
|
|
15
|
+
LetterF = PigpenSegment.North | PigpenSegment.West | PigpenSegment.South,
|
|
16
|
+
LetterG = PigpenSegment.North | PigpenSegment.East,
|
|
17
|
+
LetterH = PigpenSegment.North | PigpenSegment.West | PigpenSegment.East,
|
|
18
|
+
LetterI = PigpenSegment.North | PigpenSegment.West,
|
|
19
|
+
|
|
20
|
+
// Grid with dot (J-R)
|
|
21
|
+
LetterJ = PigpenSegment.East | PigpenSegment.South | PigpenSegment.Dot,
|
|
22
|
+
LetterK = PigpenSegment.West |
|
|
23
|
+
PigpenSegment.East |
|
|
24
|
+
PigpenSegment.South |
|
|
25
|
+
PigpenSegment.Dot,
|
|
26
|
+
LetterL = PigpenSegment.West | PigpenSegment.South | PigpenSegment.Dot,
|
|
27
|
+
LetterM = PigpenSegment.North |
|
|
28
|
+
PigpenSegment.East |
|
|
29
|
+
PigpenSegment.South |
|
|
30
|
+
PigpenSegment.Dot,
|
|
31
|
+
LetterN = PigpenSegment.North |
|
|
32
|
+
PigpenSegment.East |
|
|
33
|
+
PigpenSegment.South |
|
|
34
|
+
PigpenSegment.West |
|
|
35
|
+
PigpenSegment.Dot,
|
|
36
|
+
LetterO = PigpenSegment.North |
|
|
37
|
+
PigpenSegment.West |
|
|
38
|
+
PigpenSegment.South |
|
|
39
|
+
PigpenSegment.Dot,
|
|
40
|
+
LetterP = PigpenSegment.North | PigpenSegment.East | PigpenSegment.Dot,
|
|
41
|
+
LetterQ = PigpenSegment.North |
|
|
42
|
+
PigpenSegment.West |
|
|
43
|
+
PigpenSegment.East |
|
|
44
|
+
PigpenSegment.Dot,
|
|
45
|
+
LetterR = PigpenSegment.North | PigpenSegment.West | PigpenSegment.Dot,
|
|
46
|
+
|
|
47
|
+
// X without dot (S-V)
|
|
48
|
+
LetterS = PigpenSegment.NorthWest | PigpenSegment.NorthEast,
|
|
49
|
+
LetterT = PigpenSegment.NorthWest | PigpenSegment.SouthWest,
|
|
50
|
+
LetterU = PigpenSegment.NorthEast | PigpenSegment.SouthEast,
|
|
51
|
+
LetterV = PigpenSegment.SouthWest | PigpenSegment.SouthEast,
|
|
52
|
+
|
|
53
|
+
// X with dot (W-Z)
|
|
54
|
+
LetterW = PigpenSegment.NorthWest |
|
|
55
|
+
PigpenSegment.NorthEast |
|
|
56
|
+
PigpenSegment.Dot,
|
|
57
|
+
LetterX = PigpenSegment.NorthWest |
|
|
58
|
+
PigpenSegment.SouthWest |
|
|
59
|
+
PigpenSegment.Dot,
|
|
60
|
+
LetterY = PigpenSegment.NorthEast |
|
|
61
|
+
PigpenSegment.SouthEast |
|
|
62
|
+
PigpenSegment.Dot,
|
|
63
|
+
LetterZ = PigpenSegment.SouthWest |
|
|
64
|
+
PigpenSegment.SouthEast |
|
|
65
|
+
PigpenSegment.Dot,
|
|
66
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export {
|
|
2
|
+
canTogglePigpenSegment,
|
|
3
|
+
decodePigpenStream,
|
|
4
|
+
hasPigpenSegment,
|
|
5
|
+
isCardinal,
|
|
6
|
+
isIntercardinal,
|
|
7
|
+
lookupPigpenEncoding,
|
|
8
|
+
togglePigpenSegment,
|
|
9
|
+
} from './pigpen.js';
|
|
10
|
+
export {PigpenEncoding} from './encoding.js';
|
|
11
|
+
export {PigpenSegment} from './segment.js';
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
import {EncodingCategory} from '../common/encoding-category.js';
|
|
2
|
+
import {EncodingEntry} from '../common/encoding-entry.js';
|
|
3
|
+
import {EncodingLookupResult} from '../common/encoding-lookup-result.js';
|
|
4
|
+
import {PigpenEncoding} from './encoding.js';
|
|
5
|
+
import {PigpenSegment} from './segment.js';
|
|
6
|
+
|
|
7
|
+
// Build the lookup table once
|
|
8
|
+
const PIGPEN_ENTRIES: Array<EncodingEntry<PigpenEncoding>> = [];
|
|
9
|
+
|
|
10
|
+
function addEntry(
|
|
11
|
+
encoding: PigpenEncoding,
|
|
12
|
+
category: EncodingCategory,
|
|
13
|
+
display: string,
|
|
14
|
+
) {
|
|
15
|
+
PIGPEN_ENTRIES.push(new EncodingEntry(encoding, category, display));
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// Letters
|
|
19
|
+
addEntry(PigpenEncoding.LetterA, EncodingCategory.Letter, 'A');
|
|
20
|
+
addEntry(PigpenEncoding.LetterB, EncodingCategory.Letter, 'B');
|
|
21
|
+
addEntry(PigpenEncoding.LetterC, EncodingCategory.Letter, 'C');
|
|
22
|
+
addEntry(PigpenEncoding.LetterD, EncodingCategory.Letter, 'D');
|
|
23
|
+
addEntry(PigpenEncoding.LetterE, EncodingCategory.Letter, 'E');
|
|
24
|
+
addEntry(PigpenEncoding.LetterF, EncodingCategory.Letter, 'F');
|
|
25
|
+
addEntry(PigpenEncoding.LetterG, EncodingCategory.Letter, 'G');
|
|
26
|
+
addEntry(PigpenEncoding.LetterH, EncodingCategory.Letter, 'H');
|
|
27
|
+
addEntry(PigpenEncoding.LetterI, EncodingCategory.Letter, 'I');
|
|
28
|
+
addEntry(PigpenEncoding.LetterJ, EncodingCategory.Letter, 'J');
|
|
29
|
+
addEntry(PigpenEncoding.LetterK, EncodingCategory.Letter, 'K');
|
|
30
|
+
addEntry(PigpenEncoding.LetterL, EncodingCategory.Letter, 'L');
|
|
31
|
+
addEntry(PigpenEncoding.LetterM, EncodingCategory.Letter, 'M');
|
|
32
|
+
addEntry(PigpenEncoding.LetterN, EncodingCategory.Letter, 'N');
|
|
33
|
+
addEntry(PigpenEncoding.LetterO, EncodingCategory.Letter, 'O');
|
|
34
|
+
addEntry(PigpenEncoding.LetterP, EncodingCategory.Letter, 'P');
|
|
35
|
+
addEntry(PigpenEncoding.LetterQ, EncodingCategory.Letter, 'Q');
|
|
36
|
+
addEntry(PigpenEncoding.LetterR, EncodingCategory.Letter, 'R');
|
|
37
|
+
addEntry(PigpenEncoding.LetterS, EncodingCategory.Letter, 'S');
|
|
38
|
+
addEntry(PigpenEncoding.LetterT, EncodingCategory.Letter, 'T');
|
|
39
|
+
addEntry(PigpenEncoding.LetterU, EncodingCategory.Letter, 'U');
|
|
40
|
+
addEntry(PigpenEncoding.LetterV, EncodingCategory.Letter, 'V');
|
|
41
|
+
addEntry(PigpenEncoding.LetterW, EncodingCategory.Letter, 'W');
|
|
42
|
+
addEntry(PigpenEncoding.LetterX, EncodingCategory.Letter, 'X');
|
|
43
|
+
addEntry(PigpenEncoding.LetterY, EncodingCategory.Letter, 'Y');
|
|
44
|
+
addEntry(PigpenEncoding.LetterZ, EncodingCategory.Letter, 'Z');
|
|
45
|
+
|
|
46
|
+
/** Mask covering all cardinal segments (grid shapes). */
|
|
47
|
+
const CARDINAL_MASK =
|
|
48
|
+
PigpenSegment.North |
|
|
49
|
+
PigpenSegment.East |
|
|
50
|
+
PigpenSegment.South |
|
|
51
|
+
PigpenSegment.West;
|
|
52
|
+
|
|
53
|
+
/** Mask covering all intercardinal segments (X shapes). */
|
|
54
|
+
const INTERCARDINAL_MASK =
|
|
55
|
+
PigpenSegment.NorthEast |
|
|
56
|
+
PigpenSegment.SouthEast |
|
|
57
|
+
PigpenSegment.SouthWest |
|
|
58
|
+
PigpenSegment.NorthWest;
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Looks up a pigpen encoding in the data table.
|
|
62
|
+
*/
|
|
63
|
+
export function lookupPigpenEncoding(
|
|
64
|
+
encoding: PigpenEncoding,
|
|
65
|
+
category: EncodingCategory = EncodingCategory.All,
|
|
66
|
+
): EncodingLookupResult<PigpenEncoding> {
|
|
67
|
+
const result = new EncodingLookupResult<PigpenEncoding>();
|
|
68
|
+
|
|
69
|
+
for (const entry of PIGPEN_ENTRIES) {
|
|
70
|
+
if ((entry.category & category) !== 0) {
|
|
71
|
+
if (entry.encoding === encoding) {
|
|
72
|
+
result.exact.push(entry);
|
|
73
|
+
} else if ((entry.encoding & encoding) === encoding) {
|
|
74
|
+
result.partial.push(entry);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
return result;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Toggles a segment flag in an encoding. Returns the encoding unchanged
|
|
84
|
+
* if toggling the segment on would mix cardinal and intercardinal segments.
|
|
85
|
+
* Toggling a segment off or toggling the dot is always allowed.
|
|
86
|
+
*/
|
|
87
|
+
export function togglePigpenSegment(
|
|
88
|
+
encoding: PigpenEncoding,
|
|
89
|
+
segment: PigpenSegment,
|
|
90
|
+
): PigpenEncoding {
|
|
91
|
+
if (!canTogglePigpenSegment(encoding, segment)) {
|
|
92
|
+
return encoding;
|
|
93
|
+
}
|
|
94
|
+
return (encoding ^ segment) as PigpenEncoding;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* Checks if a segment flag is set in an encoding.
|
|
99
|
+
*/
|
|
100
|
+
export function hasPigpenSegment(
|
|
101
|
+
encoding: PigpenEncoding,
|
|
102
|
+
segment: PigpenSegment,
|
|
103
|
+
): boolean {
|
|
104
|
+
return (encoding & segment) === segment;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* Returns true if the encoding uses cardinal (grid) segments.
|
|
109
|
+
*/
|
|
110
|
+
export function isCardinal(encoding: PigpenEncoding): boolean {
|
|
111
|
+
return ((encoding as number) & CARDINAL_MASK) !== 0;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Returns true if the encoding uses intercardinal (X) segments.
|
|
116
|
+
*/
|
|
117
|
+
export function isIntercardinal(encoding: PigpenEncoding): boolean {
|
|
118
|
+
return ((encoding as number) & INTERCARDINAL_MASK) !== 0;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Checks if a segment can be toggled without mixing cardinal and intercardinal
|
|
123
|
+
* segments. Toggling off and toggling the dot are always allowed.
|
|
124
|
+
*/
|
|
125
|
+
export function canTogglePigpenSegment(
|
|
126
|
+
encoding: PigpenEncoding,
|
|
127
|
+
segment: PigpenSegment,
|
|
128
|
+
): boolean {
|
|
129
|
+
// Toggling off is always allowed
|
|
130
|
+
if (hasPigpenSegment(encoding, segment)) {
|
|
131
|
+
return true;
|
|
132
|
+
}
|
|
133
|
+
// Dot is always compatible
|
|
134
|
+
if (segment === PigpenSegment.Dot) {
|
|
135
|
+
return true;
|
|
136
|
+
}
|
|
137
|
+
// Reject masks that mix cardinal and intercardinal bits
|
|
138
|
+
const segmentAsEncoding = segment as number as PigpenEncoding;
|
|
139
|
+
const segmentHasCardinal = isCardinal(segmentAsEncoding);
|
|
140
|
+
const segmentHasIntercardinal = isIntercardinal(segmentAsEncoding);
|
|
141
|
+
if (segmentHasCardinal && segmentHasIntercardinal) {
|
|
142
|
+
return false;
|
|
143
|
+
}
|
|
144
|
+
if (segmentHasCardinal) {
|
|
145
|
+
return !isIntercardinal(encoding);
|
|
146
|
+
}
|
|
147
|
+
return !isCardinal(encoding);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* Decodes an array of pigpen encodings to a string.
|
|
152
|
+
*/
|
|
153
|
+
export function decodePigpenStream(encodings: PigpenEncoding[]): string {
|
|
154
|
+
let result = '';
|
|
155
|
+
|
|
156
|
+
for (const ch of encodings) {
|
|
157
|
+
if (ch === PigpenEncoding.None) {
|
|
158
|
+
result += ' ';
|
|
159
|
+
} else {
|
|
160
|
+
const lookup = lookupPigpenEncoding(ch, EncodingCategory.Letter);
|
|
161
|
+
if (lookup.exact.length > 0) {
|
|
162
|
+
result += lookup.exact[0].toString();
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
return result;
|
|
168
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export {
|
|
2
|
+
addSemaphoreDirection,
|
|
3
|
+
decodeSemaphoreStream,
|
|
4
|
+
degreesToSemaphoreDirection,
|
|
5
|
+
directionsToEncoding,
|
|
6
|
+
getEncodingDegrees,
|
|
7
|
+
hasSemaphoreDirection,
|
|
8
|
+
lookupSemaphoreEncoding,
|
|
9
|
+
removeSemaphoreDirection,
|
|
10
|
+
semaphoreDirectionToDegrees,
|
|
11
|
+
} from './semaphore.js';
|
|
12
|
+
export {SemaphoreDirection} from './direction.js';
|
|
13
|
+
export {SemaphoreEncoding} from './encoding.js';
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import {EncodingCategory} from '../
|
|
2
|
-
import {EncodingEntry} from '../
|
|
3
|
-
import {EncodingLookupResult} from '../
|
|
4
|
-
import {SemaphoreDirection} from './
|
|
5
|
-
import {SemaphoreEncoding} from './
|
|
1
|
+
import {EncodingCategory} from '../common/encoding-category.js';
|
|
2
|
+
import {EncodingEntry} from '../common/encoding-entry.js';
|
|
3
|
+
import {EncodingLookupResult} from '../common/encoding-lookup-result.js';
|
|
4
|
+
import {SemaphoreDirection} from './direction.js';
|
|
5
|
+
import {SemaphoreEncoding} from './encoding.js';
|
|
6
6
|
|
|
7
7
|
// Build the lookup table once
|
|
8
8
|
const SEMAPHORE_ENTRIES: Array<EncodingEntry<SemaphoreEncoding>> = [];
|
|
@@ -4,8 +4,6 @@
|
|
|
4
4
|
// Definitions: https://github.com/DefinitelyTyped/DefinitelyTyped
|
|
5
5
|
// TypeScript Version: 2.8
|
|
6
6
|
|
|
7
|
-
// The published typings don't seem to work correctly with ES6 module syntax,
|
|
8
|
-
// switch to CommonJS module syntax instead.
|
|
9
7
|
declare module 'trie-prefix-tree' {
|
|
10
8
|
function Trie(strings: string[]): {
|
|
11
9
|
/**
|
|
@@ -66,5 +64,5 @@ declare module 'trie-prefix-tree' {
|
|
|
66
64
|
getSubAnagrams(word: string): string[];
|
|
67
65
|
};
|
|
68
66
|
|
|
69
|
-
export
|
|
67
|
+
export default Trie;
|
|
70
68
|
}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export type {WordSearchPoint} from './point.js';
|
|
2
|
+
export {WordSearchResult} from './result.js';
|
|
3
|
+
export {WordSearchDirection} from './direction.js';
|
|
4
|
+
export {findWords, parseWordSearchGrid} from './word-search.js';
|
|
5
|
+
export type {WordSearchOptions} from './word-search.js';
|