1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
|
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include "strarr.h"
/* eow: guess the last three letters of a set of words.
e.g. ti___, man___, cas___: the player must guess `tle`.
How to do
- Eliminate all words less than NUMBER_OF_CHARS + 1 chars.
- Then somehow search for common endings...?
for (word in words)
ending = get ending
for (word in remaining words)
10 => 10 + 9 + 8 + 7 ...
For 370k words, that is 68 bil. That will take seconds.
I think, since this is a simple game, I will just pick
out words at random. In other words, randomly index
into the array (possibly costly in this structure)...
I think add a function to generate interstitial indexes
and perform random (not: linear from start-to-end)
indexes.
Then, iterate over the entire list (subsecond) and
find ones which share the same ending. Use those
for the game.
TODO
+ Finish logic to accept a guess.
+ Add flags to change settings.
- Make number of guesses configurable.
+ Add option to get a hint, which gives away one of the letters.
(When it does this, it reveals the letter in the actual word displayed.)
- Figure out bug that when you run /hint, it for some reason
goes prints a line up, swallowing the topmost line?
- Add in /set commands.
*/
int MIN_LENGTH = 6;
int NUMBER_OF_CHARS = 3;
int NUMBER_OF_WORDS = 3;
int NUMBER_OF_TRIES = 100;
/* Get the first entry from the words as a random index. */
char *
random_index(strarr *words_arr, int min_length)
{
for (;;) {
int idx = rand() % words_arr->len;
char *w = strarr_random_index(words_arr, idx);
if (strlen(w) < min_length) {
continue;
}
return w;
}
}
/* Get strings matching the ending of the first word.
Caller must free the array but not the members.
Returns NULL if the allocation fails, or there
are insufficient matching strings for the entry
provided.
*/
const char **
get_matching_strs(strarr *words_arr, const char *first)
{
// The `-1` is due to the first word already being chosen.
const char **arr = calloc(sizeof(char*), NUMBER_OF_WORDS - 1);
if (!arr) {
return NULL;
}
/* Get ending. */
const char *ending = first + (strlen(first) - NUMBER_OF_CHARS);
/* Iterate over words and find NUMBER_OF_WORDS entries with
this ending.
*/
char *p;
int i;
for (i = 0, p = strarr_index(words_arr, 0); p != NULL; p = strarr_next(words_arr, p)) {
size_t sz = strlen(p);
if (sz < MIN_LENGTH
|| strcmp((p + (sz - NUMBER_OF_CHARS)), ending) != 0
|| strcmp(p, first) == 0) {
continue;
}
/* If it's passed the check, we know it is the right length, and has the correct
ending: add it to the list, check if we've got all the words, quit if so.
*/
arr[i++] = p;
if (i == NUMBER_OF_WORDS - 1) {
return arr;
}
}
/* If we managed to break out of the loop, it means we've iterated over the
entire array and not found enough words. So return NULL to signify error.
*/
free(arr);
return NULL;
}
void
print_n_uscore(int i) {
for (int rep = 0; rep < i; ++rep) {
printf("_");
}
}
void
display_prompt(int totalpoints, int points, const char *first, const char **others, const char *hinted_ending, int guesses)
{
printf("Points: %d Points available: %d\n", totalpoints, points);
printf("Your words are: \n");
printf(" %.*s", (int)strlen(first) - NUMBER_OF_CHARS, first);
printf("%s", hinted_ending);
printf("\n");
for (int i = 0; i < NUMBER_OF_WORDS - 1; ++i) {
printf(" %.*s", (int)strlen(others[i]) - NUMBER_OF_CHARS, others[i]);
printf("%s", hinted_ending);
printf("\n");
}
printf("What do you think the word is (incorrect guesses: %d): ", guesses);
}
void
clear_prompt()
{
static int is_first_call = 1;
if (is_first_call) {
/* do nothing */
is_first_call = 0;
return;
}
int lines_displayed = 3 + (NUMBER_OF_WORDS - 1) + 1 + 1 + 1;
for (int c = 0; c < lines_displayed - 1; ++c) {
printf(" ");
printf("\033[F");
}
}
int
ask(const char *word, const char **others, int totalpoints)
{
const char *ending = word + strlen(word) - NUMBER_OF_CHARS;
char *hinted_ending = malloc(strlen(ending));
for (int i = 0; i < strlen(ending); ++i) {
hinted_ending[i] = '_';
}
hinted_ending[strlen(ending)] = '\0';
printf("Type /help for help, or /hint, or /giveup, or /quit to quit.\n");
int hinted_to = 0;
int guesses = 0;
int points = 5;
for (;;) {
/* Display the words. */
clear_prompt();
display_prompt(totalpoints, points, word, others, hinted_ending, guesses);
/* Get input. */
char ui[256];
scanf("%s", ui); /* TODO: add safety to check len(input) < 256 */
if (!strcmp(ui, ending)) {
break;
} else if (!strcmp(ui, "/help")) {
/* TODO add help */
} else if (!strcmp(ui, "/hint")) {
if (hinted_to == NUMBER_OF_CHARS - 1) {
continue; /* do not give hint as hints are exhausted*/
}
hinted_ending[hinted_to] = ending[hinted_to];
hinted_to++;
if (--points == 0) {
points = 1;
}
} else if (!strcmp(ui, "/giveup")) {
points = -1;
break;
} else if (!strcmp(ui, "/quit")) {
exit(0);
} else {
guesses++;
if (--points == 0) {
points = 1;
}
continue;
}
}
return points;
}
void
run(strarr *words)
{
int points = 0;
for (;;) {
/* Generate the first word, guaranteed to be at least the min length. */
const char *w = random_index(words, MIN_LENGTH);
/* Find two words with matching endings - for now, just grab the first
NUMBER_OF_WORDS entries we find, but...
TODO: add logic to get more than just the first two entries.
*/
const char **others = NULL;
/* Continually retries until a valid set found. If no valid words found
after NUMBER_OF_TRIES attempts, just panic and die. */
int i = 0;
while (i++ < NUMBER_OF_TRIES && !others) {
others = get_matching_strs(words, w);
}
if (!others) {
// printf("No words matching the ending of %s - retrying!\n", w);
continue;
}
/* Ask the user what the suffix is. */
points += ask(w, others, points);
free(others);
}
}
int main(int argc, char *argv[])
{
/* Setup. */
srand(time(NULL));
/* Parse arguments. */
for (int i = 0; i < argc; ++i) {
if (argv[i][0] == '-') {
char f = argv[i][1];
switch (f) {
case 'l':
MIN_LENGTH = strtol(argv[++i], NULL, 10);
if (MIN_LENGTH < 2) {
fprintf(stderr, "-l flag invalid\n");
exit(1);
}
break;
case 'c':
NUMBER_OF_CHARS = strtol(argv[++i], NULL, 10);
if (NUMBER_OF_CHARS < 1) {
fprintf(stderr, "-c flag invalid\n");
exit(2);
}
break;
case 'n':
NUMBER_OF_WORDS = strtol(argv[++i], NULL, 10);
if (NUMBER_OF_WORDS < 2) {
fprintf(stderr, "-n flag invalid\n");
exit(3);
}
break;
case 't':
NUMBER_OF_TRIES = strtol(argv[++i], NULL, 10);
if (NUMBER_OF_TRIES < 2) {
fprintf(stderr, "-t flag invalid\n");
exit(4);
}
break;
}
}
}
if (MIN_LENGTH <= NUMBER_OF_CHARS) {
fprintf(stderr, "min length of word (-l) cannot be less than number of characters missing (-c)\n");
exit(5);
}
/* Read contents of file. */
FILE *fp = fopen("./words.txt", "r");
fseek(fp, 0, SEEK_END);
long fsize = ftell(fp);
fseek(fp, 0, SEEK_SET);
char *words = malloc(fsize + 1);
fread(words, fsize, 1, fp);
fclose(fp);
words[fsize] = 0;
/* Convert into an array. */
// size_t words_sz = 0;
strarr words_arr;
strarr_init_from_str_split(&words_arr, words, '\n');
/* Run the game. */
run(&words_arr);
strarr_free(&words_arr);
return 0;
}
|