]>
git.cameronkatri.com Git - bsdgames-darwin.git/blob - cgram/cgram.c
1 /* $NetBSD: cgram.c,v 1.12 2021/02/22 16:28:20 rillig Exp $ */
4 * Copyright (c) 2013, 2021 The NetBSD Foundation, Inc.
7 * This code is derived from software contributed to The NetBSD Foundation
8 * by David A. Holland and Roland Illig.
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
32 #include <sys/cdefs.h>
33 #if defined(__RCSID) && !defined(lint)
34 __RCSID("$NetBSD: cgram.c,v 1.12 2021/02/22 16:28:20 rillig Exp $");
47 #include "pathnames.h"
49 ////////////////////////////////////////////////////////////
54 return (char)toupper((unsigned char)ch
);
60 return (char)tolower((unsigned char)ch
);
66 return isalpha((unsigned char)ch
) != 0;
72 return islower((unsigned char)ch
) != 0;
78 return isupper((unsigned char)ch
) != 0;
93 ////////////////////////////////////////////////////////////
107 string_init(struct string
*s
)
115 string_add(struct string
*s
, char ch
)
117 if (s
->len
>= s
->cap
) {
118 s
->cap
= 2 * s
->cap
+ 16;
119 s
->s
= realloc(s
->s
, s
->cap
);
121 errx(1, "Out of memory");
127 string_finish(struct string
*s
)
134 stringarray_init(struct stringarray
*a
)
141 stringarray_cleanup(struct stringarray
*a
)
143 for (size_t i
= 0; i
< a
->num
; i
++)
149 stringarray_add(struct stringarray
*a
, struct string
*s
)
151 size_t num
= a
->num
++;
152 a
->v
= realloc(a
->v
, a
->num
* sizeof a
->v
[0]);
154 errx(1, "Out of memory");
159 stringarray_dup(struct stringarray
*dst
, const struct stringarray
*src
)
161 assert(dst
->num
== 0);
162 for (size_t i
= 0; i
< src
->num
; i
++) {
165 for (const char *p
= src
->v
[i
].s
; *p
!= '\0'; p
++)
166 string_add(&str
, *p
);
168 stringarray_add(dst
, &str
);
172 ////////////////////////////////////////////////////////////
174 static struct stringarray lines
;
175 static struct stringarray sollines
;
187 return (int)lines
.v
[cursor_y
].len
;
199 FILE *f
= popen(_PATH_FORTUNE
, "r");
201 err(1, "%s", _PATH_FORTUNE
);
207 while ((ch
= fgetc(f
)) != EOF
) {
209 string_finish(&line
);
210 stringarray_add(&lines
, &line
);
212 } else if (ch
== '\t') {
213 string_add(&line
, ' ');
214 while (line
.len
% 8 != 0)
215 string_add(&line
, ' ');
216 } else if (ch
== '\b') {
220 string_add(&line
, (char)ch
);
224 stringarray_dup(&sollines
, &lines
);
226 extent_y
= (int)lines
.num
;
227 for (int i
= 0; i
< extent_y
; i
++)
228 extent_x
= imax(extent_x
, (int)lines
.v
[i
].len
);
231 exit(1); /* error message must come from child process */
239 for (int i
= 0; i
< 26; i
++)
242 for (int i
= 26; i
> 1; i
--) {
243 int c
= (int)(random() % i
);
249 for (int y
= 0; y
< extent_y
; y
++) {
250 for (char *p
= lines
.v
[y
].s
; *p
!= '\0'; p
++) {
252 *p
= (char)('a' + key
[*p
- 'a']);
254 *p
= (char)('A' + key
[*p
- 'A']);
262 assert(cursor_x
>= 0 && cursor_x
< extent_x
);
263 assert(cursor_y
>= 0 && cursor_y
< extent_y
);
264 if (cursor_x
>= cur_max_x()) {
269 char och
= lines
.v
[cursor_y
].s
[cursor_x
];
270 if (!ch_isalpha(och
)) {
275 char loch
= ch_tolower(och
);
276 char uoch
= ch_toupper(och
);
277 char lch
= ch_tolower(ch
);
278 char uch
= ch_toupper(ch
);
280 for (int y
= 0; y
< (int)lines
.num
; y
++) {
281 for (char *p
= lines
.v
[y
].s
; *p
!= '\0'; p
++) {
295 ////////////////////////////////////////////////////////////
300 for (size_t i
= 0; i
< lines
.num
; i
++)
301 if (strcmp(lines
.v
[i
].s
, sollines
.v
[i
].s
) != 0)
311 int max_y
= imin(LINES
- 1, extent_y
- offset_y
);
312 for (int y
= 0; y
< max_y
; y
++) {
315 int len
= (int)lines
.v
[offset_y
+ y
].len
;
316 int max_x
= imin(COLS
- 1, len
- offset_x
);
317 const char *line
= lines
.v
[offset_y
+ y
].s
;
318 const char *solline
= sollines
.v
[offset_y
+ y
].s
;
320 for (int x
= 0; x
< max_x
; x
++) {
321 char ch
= line
[offset_x
+ x
];
322 bool bold
= hinting
&&
323 ch
== solline
[offset_x
+ x
] &&
338 addstr("~ to quit, * to cheat, ^pnfb to move");
340 move(cursor_y
- offset_y
, cursor_x
- offset_x
);
351 keypad(stdscr
, true);
360 ////////////////////////////////////////////////////////////
363 saturate_cursor(void)
365 cursor_y
= imax(cursor_y
, 0);
366 cursor_y
= imin(cursor_y
, cur_max_y());
368 assert(cursor_x
>= 0);
369 cursor_x
= imin(cursor_x
, cur_max_x());
373 scroll_into_view(void)
375 if (cursor_x
< offset_x
)
377 if (cursor_x
> offset_x
+ COLS
- 1)
378 offset_x
= cursor_x
- (COLS
- 1);
380 if (cursor_y
< offset_y
)
382 if (cursor_y
> offset_y
+ LINES
- 2)
383 offset_y
= cursor_y
- (LINES
- 2);
387 handle_char_input(int ch
)
389 if (isascii(ch
) && ch_isalpha((char)ch
)) {
390 if (substitute((char)ch
)) {
391 if (cursor_x
< cur_max_x())
393 if (cursor_x
== cur_max_x() &&
394 cursor_y
< cur_max_y()) {
399 } else if (cursor_x
< cur_max_x() &&
400 ch
== lines
.v
[cursor_y
].s
[cursor_x
]) {
402 if (cursor_x
== cur_max_x() &&
403 cursor_y
< cur_max_y()) {
426 } else if (cursor_y
> 0) {
428 cursor_x
= cur_max_x();
433 cursor_x
= cur_max_x();
437 if (cursor_x
< cur_max_x()) {
439 } else if (cursor_y
< cur_max_y()) {
456 cursor_y
-= LINES
- 2;
459 cursor_y
+= LINES
- 2;
467 handle_char_input(ch
);
476 stringarray_init(&lines
);
477 stringarray_init(&sollines
);
478 srandom((unsigned int)time(NULL
));
500 stringarray_cleanup(&sollines
);
501 stringarray_cleanup(&lines
);
504 ////////////////////////////////////////////////////////////