1 /* vim:tw=78:ts=8:sw=4:set ft=c: */
3 Copyright (C) 2002-2006 Ben Kibbey <bjk@luxsci.net>
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #include <sys/types.h>
25 #include <sys/socket.h>
75 static char *str_etc(const char *str
, int maxlen
, int rev
)
77 int len
= strlen(str
);
78 static char buf
[80], *p
= buf
;
81 strncpy(buf
, str
, sizeof(buf
));
90 for (i
= 0; i
< maxlen
+ 3; i
++)
91 *p
++ = buf
[(len
- maxlen
) + i
+ 3];
107 void update_cursor(GAME g
, int idx
)
111 int t
= history_total(g
.hp
);
114 * If not deincremented then r and c would be the next move.
118 if (idx
> t
|| idx
< 0 || !t
|| !g
.hp
[idx
]->move
) {
119 c_row
= 2, c_col
= 5;
132 c_row
= (g
.turn
== WHITE
) ? 8 : 1;
141 c_row
= ROWTOINT(*p
--);
142 c_col
= COLTOINT(*p
);
145 static int init_nag()
151 if ((fp
= fopen(config
.nagfile
, "r")) == NULL
) {
152 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.nagfile
, strerror(errno
));
157 if (fscanf(fp
, " %[^\n] ", line
) == 1) {
158 nags
= Realloc(nags
, (i
+ 2) * sizeof(struct nag_s
));
159 nags
[i
].line
= strdup(line
);
169 char *history_edit_nag(void *arg
)
173 ITEM
**mitems
= NULL
;
179 HISTORY
*anno
= (HISTORY
*)arg
;
187 mitems
= Realloc(mitems
, (i
+ 2) * sizeof(ITEM
));
188 mitems
[i
++] = new_item(NONE
, NULL
);
190 for (n
= 0; nags
[n
].line
; n
++, i
++) {
191 mitems
= Realloc(mitems
, (i
+ 2) * sizeof(ITEM
));
192 mitems
[i
] = new_item(nags
[n
].line
, NULL
);
196 menu
= new_menu(mitems
);
197 scale_menu(menu
, &rows
, &cols
);
199 win
= newwin(rows
+ 4, cols
+ 2, CALCPOSY(rows
) - 2, CALCPOSX(cols
));
200 set_menu_win(menu
, win
);
201 subw
= derwin(win
, rows
, cols
, 2, 1);
202 set_menu_sub(menu
, subw
);
203 set_menu_fore(menu
, A_REVERSE
);
204 set_menu_grey(menu
, A_NORMAL
);
205 set_menu_mark(menu
, NULL
);
206 set_menu_spacing(menu
, 0, 0, 0);
207 menu_opts_off(menu
, O_NONCYCLIC
|O_SHOWDESC
|O_ONEVALUE
);
209 panel
= new_panel(win
);
213 set_menu_pattern(menu
, mbuf
);
214 wbkgd(win
, CP_MESSAGE_WINDOW
);
215 draw_window_title(win
, NAG_EDIT_TITLE
, cols
+ 2, CP_HISTORY_TITLE
,
218 for (i
= 0; i
< MAX_PGN_NAG
; i
++) {
219 if (anno
->nag
[i
] && anno
->nag
[i
] <= item_count(menu
)) {
220 set_item_value(mitems
[anno
->nag
[i
]], TRUE
);
221 set_current_item(menu
, mitems
[anno
->nag
[i
]]);
231 wattron(win
, A_REVERSE
);
233 for (c
= 1; c
< (cols
+ 2) - 1; c
++)
234 mvwprintw(win
, rows
+ 2, c
, " ");
236 c
= item_index(current_item(menu
)) + 1;
238 snprintf(buf
, sizeof(buf
), "Item %i of %i (%i of %i selected) %s", c
,
239 item_count(menu
), itemcount
, MAX_PGN_NAG
, NAG_EDIT_PROMPT
);
240 draw_prompt(win
, rows
+ 2, cols
+ 2, buf
, CP_MESSAGE_PROMPT
);
242 wattroff(win
, A_REVERSE
);
245 for (i
= 0; mitems
[i
]; i
++)
246 set_item_value(mitems
[i
], FALSE
);
248 set_item_value(mitems
[0], TRUE
);
251 set_item_value(mitems
[0], FALSE
);
253 /* This nl() statement needs to be here because NL is recognized
254 * for some reason after the first selection.
264 help(NAG_EDIT_HELP
, ANYKEY
, naghelp
);
272 for (i
= item_index(current_item(menu
)) + 1; mitems
[i
]; i
++) {
273 if (item_value(mitems
[i
]) == TRUE
) {
280 for (i
= 0; mitems
[i
]; i
++) {
281 if (item_value(mitems
[i
]) == TRUE
) {
288 set_current_item(menu
, mitems
[found
]);
296 for (i
= item_index(current_item(menu
)) - 1; i
> 0; i
--) {
297 if (item_value(mitems
[i
]) == TRUE
) {
304 for (i
= item_count(menu
) - 1; i
> 0; i
--) {
305 if (item_value(mitems
[i
]) == TRUE
) {
312 set_current_item(menu
, mitems
[found
]);
315 menu_driver(menu
, REQ_FIRST_ITEM
);
318 menu_driver(menu
, REQ_LAST_ITEM
);
321 menu_driver(menu
, REQ_UP_ITEM
);
324 menu_driver(menu
, REQ_DOWN_ITEM
);
328 if (menu_driver(menu
, REQ_SCR_UPAGE
) == E_REQUEST_DENIED
)
329 menu_driver(menu
, REQ_FIRST_ITEM
);
333 if (menu_driver(menu
, REQ_SCR_DPAGE
) == E_REQUEST_DENIED
)
334 menu_driver(menu
, REQ_LAST_ITEM
);
337 if (item_index(current_item(menu
)) == 0 &&
338 item_value(current_item(menu
)) == FALSE
) {
343 if (item_value(current_item(menu
)) == TRUE
) {
344 set_item_value(current_item(menu
), FALSE
);
348 if (itemcount
+ 1 > MAX_PGN_NAG
)
351 set_item_value(current_item(menu
), TRUE
);
355 SET_FLAG(game
[gindex
].flags
, GF_MODIFIED
);
364 tmp
= menu_pattern(menu
);
366 if (tmp
&& tmp
[strlen(tmp
) - 1] != c
) {
367 menu_driver(menu
, REQ_CLEAR_PATTERN
);
368 menu_driver(menu
, c
);
371 if (menu_driver(menu
, REQ_NEXT_MATCH
) == E_NO_MATCH
)
372 menu_driver(menu
, c
);
380 for (i
= 0; i
< MAX_PGN_NAG
; i
++)
383 for (i
= 0, n
= 0; mitems
[i
] && n
< MAX_PGN_NAG
; i
++) {
384 if (item_value(mitems
[i
]) == TRUE
)
392 for (i
= 0; mitems
[i
]; i
++)
393 free_item(mitems
[i
]);
402 static void view_nag(void *arg
)
404 HISTORY
*h
= (HISTORY
*)arg
;
406 char line
[LINE_MAX
] = {0};
409 snprintf(buf
, sizeof(buf
), "Viewing NAG for \"%s\"", h
->move
);
416 for (i
= 0; i
< MAX_PGN_NAG
; i
++) {
420 strncat(line
, nags
[h
->nag
[i
] - 1].line
, sizeof(line
));
421 strncat(line
, "\n", sizeof(line
));
424 line
[strlen(line
) - 1] = 0;
425 message(buf
, ANYKEY
, "%s", line
);
428 void view_annotation(HISTORY h
)
430 char buf
[MAX_SAN_MOVE_LEN
+ strlen(ANNOTATION_VIEW_TITLE
) + 4];
431 int nag
= 0, comment
= 0;
433 if (h
.comment
&& h
.comment
[0])
439 if (!nag
&& !comment
)
442 snprintf(buf
, sizeof(buf
), "%s \"%s\"", ANNOTATION_VIEW_TITLE
, h
.move
);
445 show_message(buf
, (nag
) ? "Any other key to continue" : ANYKEY
,
446 (nag
) ? "Press 'n' to view NAG" : NULL
,
447 (nag
) ? view_nag
: NULL
, (nag
) ? (void *)&h
: NULL
,
448 (nag
) ? 'n' : 0, "%s", h
.comment
);
450 show_message(buf
, "Any other key to continue", "Press 'n' to view NAG",
451 view_nag
, (void *)&h
, 'n', "%s", "No annotations for this move");
454 static void cleanup(WINDOW
*win
, WINDOW
*subw
, PANEL
*panel
, MENU
*menu
,
455 ITEM
**items
, struct d_entries
*entries
)
462 for (i
= 0; items
[i
]; i
++)
468 for (i
= 0; entries
[i
].name
; i
++) {
469 free(entries
[i
].name
);
470 free(entries
[i
].fancy
);
481 static int sort_entries(const void *s1
, const void *s2
)
483 const struct d_entries
*ss1
= s1
;
484 const struct d_entries
*ss2
= s2
;
486 return strcmp(ss1
->name
, ss2
->name
);
489 char *browse_directory(void *arg
)
491 char *inputstr
= (char *)arg
;
492 int initkey
= (inputstr
) ? inputstr
[0] : 0;
493 char pattern
[FILENAME_MAX
];
494 static char path
[FILENAME_MAX
];
495 static char file
[FILENAME_MAX
];
500 if (config
.savedirectory
) {
501 if ((p
= word_expand(config
.savedirectory
)) == NULL
)
504 strncpy(path
, p
, sizeof(path
));
506 if (access(path
, R_OK
) == -1) {
507 cmessage(ERROR
, ANYKEY
, "%s: %s", path
, strerror(errno
));
508 getcwd(path
, sizeof(path
));
512 getcwd(path
, sizeof(path
));
517 * First find directories (including hidden) in the working directory.
518 * Then apply the config.pattern to regular files.
520 if ((p
= word_split_append(path
, '/', ".* *")) == NULL
)
523 strncpy(pattern
, p
, sizeof(pattern
));
528 ITEM
**mitems
= NULL
;
535 int len
= strlen(path
);
538 struct d_entries
*entries
= NULL
;
543 if (wordexp(pattern
, &w
, x
) != 0) {
544 cmessage(ERROR
, ANYKEY
, "Error in pattern\n%s", pattern
);
548 for (i
= 0; i
< w
.we_wordc
; i
++) {
552 if (stat(w
.we_wordv
[i
], &st
) == -1)
555 if ((p
= strrchr(w
.we_wordv
[i
], '/')) != NULL
)
561 if (!S_ISDIR(st
.st_mode
))
564 if (p
[0] == '.' && p
[1] == 0)
568 if (S_ISDIR(st
.st_mode
))
573 entries
= Realloc(entries
, (n
+ 2) * sizeof(struct d_entries
));
574 entries
[n
].name
= strdup(w
.we_wordv
[i
]);
575 entries
[n
].fancy
= Malloc(len
);
576 strncpy(entries
[n
].fancy
, p
, len
);
578 if (S_ISDIR(st
.st_mode
))
579 entries
[n
].fancy
[len
- 2] = '/';
581 tp
= localtime(&st
.st_mtime
);
582 strftime(tbuf
, sizeof(tbuf
), "%b %d %T", tp
);
584 snprintf(entries
[n
].desc
, sizeof(entries
[n
].desc
), "%-7i %s",
585 (int)st
.st_size
, tbuf
);
587 memset(&entries
[++n
], '\0', sizeof(struct d_entries
));
593 if ((p
= word_split_append(path
, '/', config
.pattern
)) == NULL
)
596 strncpy(pattern
, p
, sizeof(pattern
));
602 qsort(entries
, n
, sizeof(struct d_entries
), sort_entries
);
604 for (i
= 0; i
< n
; i
++) {
605 mitems
= Realloc(mitems
, (idx
+ 2) * sizeof(ITEM
));
606 mitems
[idx
++] = new_item(entries
[i
].fancy
, entries
[i
].desc
);
610 menu
= new_menu(mitems
);
611 scale_menu(menu
, &rows
, &cols
);
613 if (cols
< strlen(path
))
616 if (cols
< strlen(HELP_PROMPT
))
617 cols
= strlen(HELP_PROMPT
);
619 rows
= (LINES
/ 5) * 4;
622 win
= newwin(rows
+ 4, cols
, CALCPOSY(rows
) - 2, CALCPOSX(cols
));
623 set_menu_format(menu
, rows
, 0);
624 set_menu_win(menu
, win
);
625 subw
= derwin(win
, rows
, cols
- 2, 2, 1);
626 set_menu_sub(menu
, subw
);
627 set_menu_fore(menu
, A_REVERSE
);
628 set_menu_grey(menu
, A_NORMAL
);
629 set_menu_mark(menu
, NULL
);
630 set_menu_spacing(menu
, 2, 0, 0);
631 menu_opts_off(menu
, O_NONCYCLIC
);
633 panel
= new_panel(win
);
635 draw_window_title(win
, path
, cols
, CP_MESSAGE_TITLE
, CP_MESSAGE_BORDER
);
636 draw_prompt(win
, rows
+ 2, cols
, HELP_PROMPT
, CP_MESSAGE_PROMPT
);
641 set_menu_pattern(menu
, mbuf
);
643 if (isgraph(initkey
)) {
644 menu_driver(menu
, initkey
);
651 /* This nl() statement needs to be here because NL is recognized
652 * for some reason after the first selection.
661 menu_driver(menu
, REQ_SCR_UPAGE
);
666 menu_driver(menu
, REQ_SCR_DPAGE
);
669 menu_driver(menu
, REQ_UP_ITEM
);
672 menu_driver(menu
, REQ_DOWN_ITEM
);
675 selected
= item_index(current_item(menu
));
679 cleanup(win
, subw
, panel
, menu
, mitems
, entries
);
684 help(BROWSER_HELP
, ANYKEY
, file_browser_help
);
687 strncpy(path
, "~/", sizeof(path
));
688 cleanup(win
, subw
, panel
, menu
, mitems
, entries
);
692 if ((tmp
= get_input_str_clear(BROWSER_CHDIR_TITLE
, NULL
))
696 strncpy(path
, tmp
, sizeof(path
));
697 cleanup(win
, subw
, panel
, menu
, mitems
, entries
);
701 tmp
= menu_pattern(menu
);
703 if (tmp
&& tmp
[strlen(tmp
) - 1] != c
) {
704 menu_driver(menu
, REQ_CLEAR_PATTERN
);
705 menu_driver(menu
, c
);
708 if (menu_driver(menu
, REQ_NEXT_MATCH
) == E_NO_MATCH
)
709 menu_driver(menu
, c
);
717 strncpy(file
, entries
[selected
].name
, sizeof(file
));
718 cleanup(win
, subw
, panel
, menu
, mitems
, entries
);
720 if (stat(file
, &st
) == -1) {
721 cmessage(ERROR
, ANYKEY
, "%s\n%s", file
, strerror(errno
));
725 if (S_ISDIR(st
.st_mode
)) {
726 p
= file
+ strlen(file
) - 2;
728 if (strcmp(p
, "..") == 0) {
729 p
= file
+ strlen(file
) - 3;
732 if ((p
= strrchr(file
, '/')) != NULL
)
733 file
[strlen(file
) - strlen(p
)] = 0;
736 strncpy(path
, file
, sizeof(path
));
740 if (S_ISREG(st
.st_mode
))
743 cmessage(ERROR
, ANYKEY
, "%s\n%s", file
, E_NOTAREGFILE
);
747 return (*file
) ? file
: NULL
;
749 static int init_country_codes()
752 char line
[LINE_MAX
], *s
;
755 if ((fp
= fopen(config
.ccfile
, "r")) == NULL
) {
756 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.ccfile
, strerror(errno
));
760 while ((s
= fgets(line
, sizeof(line
), fp
)) != NULL
) {
763 if ((tmp
= strsep(&s
, " ")) == NULL
)
772 ccodes
= Realloc(ccodes
, (cindex
+ 2) * sizeof(struct country_codes
));
773 strncpy(ccodes
[cindex
].code
, tmp
, sizeof(ccodes
[cindex
].code
));
774 strncpy(ccodes
[cindex
].country
, s
, sizeof(ccodes
[cindex
].country
));
778 memset(&ccodes
[cindex
], '\0', sizeof(struct country_codes
));
784 char *country_codes(void *arg
)
788 ITEM
**mitems
= NULL
;
796 if (init_country_codes())
800 for (n
= i
= 0; ccodes
[n
].code
[0]; n
++, i
++) {
801 mitems
= Realloc(mitems
, (i
+ 2) * sizeof(ITEM
));
802 mitems
[i
] = new_item(ccodes
[n
].country
, ccodes
[n
].code
);
806 menu
= new_menu(mitems
);
807 scale_menu(menu
, &rows
, &cols
);
809 if (cols
< strlen(HELP_PROMPT
) + 21)
810 cols
= strlen(HELP_PROMPT
) + 21;
812 win
= newwin(rows
+ 4, cols
+ 4, CALCPOSY(rows
) - 2, CALCPOSX(cols
));
813 set_menu_win(menu
, win
);
814 subw
= derwin(win
, rows
, cols
+ 2, 2, 1);
815 set_menu_sub(menu
, subw
);
816 set_menu_fore(menu
, A_REVERSE
);
817 set_menu_grey(menu
, A_NORMAL
);
818 set_menu_mark(menu
, NULL
);
819 set_menu_spacing(menu
, 0, 0, 0);
820 menu_opts_off(menu
, O_NONCYCLIC
);
822 panel
= new_panel(win
);
826 set_menu_pattern(menu
, mbuf
);
827 wbkgd(win
, CP_MESSAGE_WINDOW
);
828 draw_window_title(win
, CC_TITLE
, cols
+ 4, CP_MESSAGE_TITLE
,
835 wattron(win
, A_REVERSE
);
837 for (c
= 1; c
< (cols
+ 2) - 1; c
++)
838 mvwprintw(win
, rows
+ 2, c
, " ");
840 c
= item_index(current_item(menu
)) + 1;
842 snprintf(buf
, sizeof(buf
), "%s %i %s %i %s", MENU_ITEM_STR
, c
,
843 N_OF_N_STR
, item_count(menu
), HELP_PROMPT
);
844 draw_prompt(win
, rows
+ 2, cols
+ 2, buf
, CP_MESSAGE_PROMPT
);
846 wattroff(win
, A_REVERSE
);
848 /* This nl() statement needs to be here because NL is recognized
849 * for some reason after the first selection.
857 help(CC_KEY_HELP
, ANYKEY
, cc_help
);
860 menu_driver(menu
, REQ_FIRST_ITEM
);
863 menu_driver(menu
, REQ_LAST_ITEM
);
866 menu_driver(menu
, REQ_UP_ITEM
);
869 menu_driver(menu
, REQ_DOWN_ITEM
);
873 if (menu_driver(menu
, REQ_SCR_UPAGE
) == E_REQUEST_DENIED
)
874 menu_driver(menu
, REQ_FIRST_ITEM
);
879 if (menu_driver(menu
, REQ_SCR_DPAGE
) == E_REQUEST_DENIED
)
880 menu_driver(menu
, REQ_LAST_ITEM
);
883 tmp
= (char *)item_description(current_item(menu
));
891 tmp
= menu_pattern(menu
);
893 if (tmp
&& tmp
[strlen(tmp
) - 1] != c
) {
894 menu_driver(menu
, REQ_CLEAR_PATTERN
);
895 menu_driver(menu
, c
);
898 if (menu_driver(menu
, REQ_NEXT_MATCH
) == E_NO_MATCH
)
899 menu_driver(menu
, c
);
910 for (i
= 0; mitems
[i
]; i
++)
911 free_item(mitems
[i
]);
919 static void add_custom_tags(TAG
***t
)
926 for (i
= 0; config
.tag
[i
]; i
++)
927 pgn_add_tag(t
, config
.tag
[i
]->name
, config
.tag
[i
]->value
);
932 TAG
**edit_tags(GAME g
, BOARD b
, int edit
)
936 unsigned char data_index
= 0;
937 int n
, lastindex
= 0;
940 /* Edit the backup copy, not the original in case the save fails. */
941 for (n
= 0; g
.tag
[n
]; n
++)
942 pgn_add_tag(&data
, g
.tag
[n
]->name
, g
.tag
[n
]->value
);
944 data_index
= pgn_tag_total(data
);
949 ITEM
**mitems
= NULL
;
957 int nlen
= 0, vlen
= 0;
959 data_index
= pgn_tag_total(data
);
961 for (i
= 0; i
< data_index
; i
++) {
962 mitems
= Realloc(mitems
, (i
+ 2) * sizeof(ITEM
));
964 if (data
[i
]->value
) {
965 nlen
= strlen(data
[i
]->name
);
966 vlen
= strlen(data
[i
]->value
);
968 /* The +6 is for the menu padding. */
969 mitems
[i
] = new_item(data
[i
]->name
,
970 (nlen
+ vlen
+ 6 >= MAX_VALUE_WIDTH
)
971 ? PRESS_ENTER
: data
[i
]->value
);
974 mitems
[i
] = new_item(data
[i
]->name
, UNKNOWN
);
978 menu
= new_menu(mitems
);
979 scale_menu(menu
, &rows
, &cols
);
981 /* +14 for the extra prompt info. */
982 if (cols
< strlen(HELP_PROMPT
) + 14)
983 cols
= strlen(HELP_PROMPT
) + 14;
985 win
= newwin(rows
+ 4, cols
+ 4, CALCPOSY(rows
) - 2, CALCPOSX(cols
));
986 set_menu_win(menu
, win
);
987 subw
= derwin(win
, rows
, cols
+ 2, 2, 1);
988 set_menu_sub(menu
, subw
);
989 set_menu_fore(menu
, A_REVERSE
);
990 set_menu_grey(menu
, A_NORMAL
);
991 set_menu_mark(menu
, NULL
);
992 set_menu_pad(menu
, '-');
993 set_menu_spacing(menu
, 3, 0, 0);
994 menu_opts_off(menu
, O_NONCYCLIC
);
996 panel
= new_panel(win
);
1001 set_menu_pattern(menu
, mbuf
);
1002 wbkgd(win
, CP_MESSAGE_WINDOW
);
1003 draw_window_title(win
, (edit
) ? TAG_EDIT_TITLE
: TAG_VIEW_TITLE
,
1004 cols
+ 4, CP_MESSAGE_TITLE
, CP_MESSAGE_BORDER
);
1008 TAG
**tmppgn
= NULL
;
1009 char *newtag
= NULL
;
1011 if (set_current_item(menu
, mitems
[lastindex
]) != E_OK
) {
1012 lastindex
= item_count(menu
) - 1;
1016 snprintf(buf
, sizeof(buf
), "%s %i %s %i %s", MENU_TAG_STR
,
1017 item_index(current_item(menu
)) + 1, N_OF_N_STR
,
1018 item_count(menu
), HELP_PROMPT
);
1019 draw_prompt(win
, rows
+ 2, cols
+ 4, buf
, CP_MESSAGE_PROMPT
);
1025 add_custom_tags(&data
);
1030 help(TAG_EDIT_HELP
, ANYKEY
, pgn_edit_help
);
1032 help(TAG_VIEW_HELP
, ANYKEY
, pgn_info_help
);
1038 selected
= item_index(current_item(menu
));
1040 if (selected
<= 6) {
1041 cmessage(NULL
, ANYKEY
, "%s", E_REMOVE_STR
);
1045 data_index
= pgn_tag_total(data
);
1047 for (i
= 0; i
< data_index
; i
++) {
1051 pgn_add_tag(&tmppgn
, data
[i
]->name
, data
[i
]->value
);
1057 for (i
= 0; tmppgn
[i
]; i
++)
1058 pgn_add_tag(&data
, tmppgn
[i
]->name
, tmppgn
[i
]->value
);
1060 pgn_tag_free(tmppgn
);
1067 if ((newtag
= get_input(TAG_NEW_TITLE
, NULL
, 1, 1, NULL
,
1068 NULL
, NULL
, 0, FIELD_TYPE_PGN_TAG_NAME
))
1072 newtag
[0] = toupper(newtag
[0]);
1074 if (strlen(newtag
) > MAX_VALUE_WIDTH
- 6 -
1075 strlen(PRESS_ENTER
)) {
1076 cmessage(ERROR
, ANYKEY
, "%s", E_TAG_NAMETOOLONG
);
1080 for (i
= 0; i
< data_index
; i
++) {
1081 if (strcasecmp(data
[i
]->name
, newtag
) == 0) {
1087 pgn_add_tag(&data
, newtag
, NULL
);
1088 data_index
= pgn_tag_total(data
);
1089 selected
= data_index
- 1;
1093 menu_driver(menu
, REQ_FIRST_ITEM
);
1096 menu_driver(menu
, REQ_LAST_ITEM
);
1102 pgn_add_tag(&data
, "FEN", pgn_game_to_fen(g
, b
));
1103 data_index
= pgn_tag_total(data
);
1104 selected
= data_index
- 1;
1109 if (menu_driver(menu
, REQ_SCR_DPAGE
) == E_REQUEST_DENIED
)
1110 menu_driver(menu
, REQ_LAST_ITEM
);
1114 if (menu_driver(menu
, REQ_SCR_UPAGE
) == E_REQUEST_DENIED
)
1115 menu_driver(menu
, REQ_FIRST_ITEM
);
1118 menu_driver(menu
, REQ_UP_ITEM
);
1121 menu_driver(menu
, REQ_DOWN_ITEM
);
1124 selected
= item_index(current_item(menu
));
1128 cleanup(win
, subw
, panel
, menu
, mitems
, NULL
);
1132 tmp
= menu_pattern(menu
);
1134 if (tmp
&& tmp
[strlen(tmp
) - 1] != c
) {
1135 menu_driver(menu
, REQ_CLEAR_PATTERN
);
1136 menu_driver(menu
, c
);
1139 if (menu_driver(menu
, REQ_NEXT_MATCH
) == E_NO_MATCH
)
1140 menu_driver(menu
, c
);
1146 lastindex
= item_index(current_item(menu
));
1150 lastindex
= selected
;
1151 nlen
= strlen(data
[selected
]->name
) + 3;
1152 nlen
+= (edit
) ? strlen(TAG_EDIT_TAG_TITLE
) : strlen(TAG_VIEW_TAG_TITLE
);
1154 if (nlen
> MAX_VALUE_WIDTH
)
1155 snprintf(buf
, sizeof(buf
), "%s", data
[selected
]->name
);
1157 snprintf(buf
, sizeof(buf
), "%s \"%s\"",
1158 (edit
) ? TAG_EDIT_TAG_TITLE
: TAG_VIEW_TAG_TITLE
,
1159 data
[selected
]->name
);
1162 if (strcmp(item_description(mitems
[selected
]), UNKNOWN
) == 0)
1165 cmessage(buf
, ANYKEY
, "%s", data
[selected
]->value
);
1169 if (strcmp(data
[selected
]->name
, "Date") == 0) {
1170 tmp
= get_input(buf
, data
[selected
]->value
, 0, 0, NULL
, NULL
, NULL
,
1171 0, FIELD_TYPE_PGN_DATE
);
1174 if (strptime(tmp
, PGN_TIME_FORMAT
, &tp
) == NULL
) {
1175 cmessage(ERROR
, ANYKEY
, "%s", E_TAG_DATE_FMT
);
1182 else if (strcmp(data
[selected
]->name
, "Site") == 0) {
1183 tmp
= get_input(buf
, data
[selected
]->value
, 1, 1, CC_PROMPT
,
1184 country_codes
, NULL
, CTRL('t'), -1);
1189 else if (strcmp(data
[selected
]->name
, "Round") == 0) {
1190 tmp
= get_input(buf
, NULL
, 1, 1, NULL
, NULL
, NULL
, 0,
1191 FIELD_TYPE_PGN_ROUND
);
1200 else if (strcmp(data
[selected
]->name
, "Result") == 0) {
1201 tmp
= get_input(buf
, data
[selected
]->value
, 1, 1, NULL
, NULL
, NULL
,
1208 if (item_description(mitems
[selected
]) &&
1209 strcmp(item_description(mitems
[selected
]), UNKNOWN
) == 0)
1212 tmp
= data
[selected
]->value
;
1214 tmp
= get_input(buf
, tmp
, 0, 0, NULL
, NULL
, NULL
, 0, -1);
1217 len
= (tmp
) ? strlen(tmp
) + 1 : 1;
1218 data
[selected
]->value
= Realloc(data
[selected
]->value
, len
);
1219 strncpy(data
[selected
]->value
, (tmp
) ? tmp
: "", len
);
1222 cleanup(win
, subw
, panel
, menu
, mitems
, NULL
);
1234 /* If the saveindex argument is -1, all games will be saved. Otherwise it's a
1235 * game index number.
1237 int save_pgn(const char *filename
, int isfifo
, int saveindex
)
1242 char buf
[FILENAME_MAX
];
1245 char *command
= NULL
;
1246 int saveindex_max
= (saveindex
== -1) ? gtotal
: saveindex
+ 1;
1248 if (filename
[0] != '/' && config
.savedirectory
&& !isfifo
) {
1249 if (stat(config
.savedirectory
, &st
) == -1) {
1250 if (errno
== ENOENT
) {
1251 if (mkdir(config
.savedirectory
, 0755) == -1) {
1252 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.savedirectory
,
1258 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.savedirectory
,
1264 stat(config
.savedirectory
, &st
);
1266 if (!S_ISDIR(st
.st_mode
)) {
1267 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.savedirectory
, E_NOTADIR
);
1271 snprintf(buf
, sizeof(buf
), "%s/%s", config
.savedirectory
, filename
);
1275 /* This is a hack to resume an existing game when more than one game is
1276 * available. Also resuming a saved game and a game from history.
1278 // FIXME: may not need this when a FEN tag is supported (by the engine).
1282 if (access(filename
, W_OK
) == 0) {
1283 c
= cmessage(NULL
, GAME_SAVE_OVERWRITE_PROMPT
,
1284 "%s \"%s\"", E_FILEEXISTS
, filename
);
1288 if (pgn_is_compressed(filename
)) {
1289 cmessage(NULL
, ANYKEY
, "%s", E_SAVE_COMPRESS
);
1307 if ((fp
= popen(command
, "w")) == NULL
) {
1308 cmessage(ERROR
, ANYKEY
, "%s: %s", filename
, strerror(errno
));
1313 if ((fp
= fopen(filename
, mode
)) == NULL
) {
1314 cmessage(ERROR
, ANYKEY
, "%s: %s", filename
, strerror(errno
));
1320 pgn_write(fp
, game
[saveindex
]);
1322 for (i
= (saveindex
== -1) ? 0 : saveindex
; i
< saveindex_max
; i
++)
1323 pgn_write(fp
, game
[i
]);
1331 if (!isfifo
&& saveindex
== -1)
1332 strncpy(loadfile
, filename
, sizeof(loadfile
));
1337 char *random_agony(GAME g
)
1341 char line
[LINE_MAX
];
1343 if (n
== -1 || !config
.agony
|| !curses_initialized
||
1344 (g
.mode
== MODE_HISTORY
&& !config
.historyagony
))
1348 if ((fp
= fopen(config
.agonyfile
, "r")) == NULL
) {
1350 cmessage(ERROR
, ANYKEY
, "%s: %s", config
.agonyfile
, strerror(errno
));
1355 if (fscanf(fp
, " %[^\n] ", line
) == 1) {
1356 agony
= Realloc(agony
, (n
+ 2) * sizeof(char *));
1357 agony
[n
++] = strdup(trim(line
));
1364 if (agony
[0] == NULL
|| !n
) {
1370 return agony
[random() % n
];
1373 static int castling_state(GAME
*g
, BOARD b
, int row
, int col
, int piece
, int mod
)
1375 if (pgn_piece_to_int(piece
) == ROOK
&& col
== 7
1377 (TEST_FLAG(g
->flags
, GF_WK_CASTLE
) || mod
) &&
1378 pgn_piece_to_int(b
[7][4].icon
) == KING
&& isupper(piece
)) {
1380 TOGGLE_FLAG(g
->flags
, GF_WK_CASTLE
);
1383 else if (pgn_piece_to_int(piece
) == ROOK
&& col
== 0
1385 (TEST_FLAG(g
->flags
, GF_WQ_CASTLE
) || mod
) &&
1386 pgn_piece_to_int(b
[7][4].icon
) == KING
&& isupper(piece
)) {
1388 TOGGLE_FLAG(g
->flags
, GF_WQ_CASTLE
);
1391 else if (pgn_piece_to_int(piece
) == ROOK
&& col
== 7
1393 (TEST_FLAG(g
->flags
, GF_BK_CASTLE
) || mod
) &&
1394 pgn_piece_to_int(b
[0][4].icon
) == KING
&& islower(piece
)) {
1396 TOGGLE_FLAG(g
->flags
, GF_BK_CASTLE
);
1399 else if (pgn_piece_to_int(piece
) == ROOK
&& col
== 0
1401 (TEST_FLAG(g
->flags
, GF_BQ_CASTLE
) || mod
) &&
1402 pgn_piece_to_int(b
[0][4].icon
) == KING
&& islower(piece
)) {
1404 TOGGLE_FLAG(g
->flags
, GF_BQ_CASTLE
);
1407 else if (pgn_piece_to_int(piece
) == KING
&& col
== 4
1409 (mod
|| (pgn_piece_to_int(b
[7][7].icon
) == ROOK
&&
1410 TEST_FLAG(g
->flags
, GF_WK_CASTLE
))
1412 (pgn_piece_to_int(b
[7][0].icon
) == ROOK
&&
1413 TEST_FLAG(g
->flags
, GF_WQ_CASTLE
))) && isupper(piece
)) {
1415 if (TEST_FLAG(g
->flags
, GF_WK_CASTLE
) ||
1416 TEST_FLAG(g
->flags
, GF_WQ_CASTLE
))
1417 CLEAR_FLAG(g
->flags
, GF_WK_CASTLE
|GF_WQ_CASTLE
);
1419 SET_FLAG(g
->flags
, GF_WK_CASTLE
|GF_WQ_CASTLE
);
1423 else if (pgn_piece_to_int(piece
) == KING
&& col
== 4
1425 (mod
|| (pgn_piece_to_int(b
[0][7].icon
) == ROOK
&&
1426 TEST_FLAG(g
->flags
, GF_BK_CASTLE
))
1428 (pgn_piece_to_int(b
[0][0].icon
) == ROOK
&&
1429 TEST_FLAG(g
->flags
, GF_BQ_CASTLE
))) && islower(piece
)) {
1431 if (TEST_FLAG(g
->flags
, GF_BK_CASTLE
) ||
1432 TEST_FLAG(g
->flags
, GF_BQ_CASTLE
))
1433 CLEAR_FLAG(g
->flags
, GF_BK_CASTLE
|GF_BQ_CASTLE
);
1435 SET_FLAG(g
->flags
, GF_BK_CASTLE
|GF_BQ_CASTLE
);
1443 static void draw_board(GAME
*g
, int details
)
1446 int bcol
= 0, brow
= 0;
1447 int maxy
= BOARD_HEIGHT
, maxx
= BOARD_WIDTH
;
1448 int ncols
= 0, offset
= 1;
1449 unsigned coords_y
= 8;
1451 for (row
= 0; row
< maxy
; row
++) {
1454 for (col
= 0; col
< maxx
; col
++) {
1457 unsigned char piece
;
1459 if (row
== 0 || row
== maxy
- 2) {
1461 mvwaddch(boardw
, row
, col
,
1462 LINE_GRAPHIC((row
) ?
1463 ACS_LLCORNER
| CP_BOARD_GRAPHICS
:
1464 ACS_ULCORNER
| CP_BOARD_GRAPHICS
));
1465 else if (col
== maxx
- 2)
1466 mvwaddch(boardw
, row
, col
,
1467 LINE_GRAPHIC((row
) ?
1468 ACS_LRCORNER
| CP_BOARD_GRAPHICS
:
1469 ACS_URCORNER
| CP_BOARD_GRAPHICS
));
1470 else if (!(col
% 4))
1471 mvwaddch(boardw
, row
, col
,
1472 LINE_GRAPHIC((row
) ?
1473 ACS_BTEE
| CP_BOARD_GRAPHICS
:
1474 ACS_TTEE
| CP_BOARD_GRAPHICS
));
1476 if (col
!= maxx
- 1)
1477 mvwaddch(boardw
, row
, col
,
1478 LINE_GRAPHIC(ACS_HLINE
| CP_BOARD_GRAPHICS
));
1484 if ((row
% 2) && col
== maxx
- 1 && coords_y
) {
1485 wattron(boardw
, CP_BOARD_COORDS
);
1486 mvwprintw(boardw
, row
, col
, "%d", coords_y
--);
1487 wattroff(boardw
, CP_BOARD_COORDS
);
1491 if ((col
== 0 || col
== maxx
- 2) && row
!= maxy
- 1) {
1493 mvwaddch(boardw
, row
, col
,
1494 LINE_GRAPHIC((col
) ?
1495 ACS_RTEE
| CP_BOARD_GRAPHICS
:
1496 ACS_LTEE
| CP_BOARD_GRAPHICS
));
1498 mvwaddch(boardw
, row
, col
,
1499 LINE_GRAPHIC(ACS_VLINE
| CP_BOARD_GRAPHICS
));
1504 if ((row
% 2) && !(col
% 4) && row
!= maxy
- 1) {
1505 mvwaddch(boardw
, row
, col
,
1506 LINE_GRAPHIC(ACS_VLINE
| CP_BOARD_GRAPHICS
));
1510 if (!(col
% 4) && row
!= maxy
- 1) {
1511 mvwaddch(boardw
, row
, col
,
1512 LINE_GRAPHIC(ACS_PLUS
| CP_BOARD_GRAPHICS
));
1523 if (((ncols
% 2) && !(offset
% 2)) || (!(ncols
% 2)
1529 if (config
.validmoves
&& g
->b
[brow
][bcol
].valid
) {
1530 attrs
= (attrwhich
== WHITE
) ? CP_BOARD_MOVES_WHITE
:
1531 CP_BOARD_MOVES_BLACK
;
1534 attrs
= (attrwhich
== WHITE
) ? CP_BOARD_WHITE
:
1537 if (row
== ROWTOMATRIX(c_row
) && col
==
1538 COLTOMATRIX(c_col
)) {
1539 attrs
= CP_BOARD_CURSOR
;
1542 if (row
== ROWTOMATRIX(sp
.row
) &&
1543 col
== COLTOMATRIX(sp
.col
)) {
1544 attrs
= CP_BOARD_SELECTED
;
1547 if (row
== maxy
- 1)
1550 mvwaddch(boardw
, row
, col
, ' ' | attrs
);
1552 if (row
== maxy
- 1)
1553 waddch(boardw
, x_grid_chars
[bcol
] | CP_BOARD_COORDS
);
1555 if (details
&& g
->b
[row
/ 2][bcol
].enpassant
)
1558 piece
= g
->b
[row
/ 2][bcol
].icon
;
1560 if (details
&& castling_state(g
, g
->b
, brow
, bcol
,
1564 if (g
->side
== WHITE
&& isupper(piece
))
1566 else if (g
->side
== BLACK
&& islower(piece
))
1569 waddch(boardw
, (pgn_piece_to_int(piece
) != OPEN_SQUARE
) ? piece
| attrs
: ' ' | attrs
);
1571 CLEAR_FLAG(attrs
, A_BOLD
);
1572 CLEAR_FLAG(attrs
, A_REVERSE
);
1575 waddch(boardw
, ' ' | attrs
);
1581 if (col
!= maxx
- 1)
1582 mvwaddch(boardw
, row
, col
,
1583 LINE_GRAPHIC(ACS_HLINE
| CP_BOARD_GRAPHICS
));
1591 void invalid_move(int n
, const char *m
)
1593 if (curses_initialized
)
1594 cmessage(ERROR
, ANYKEY
, "%s \"%s\" (round #%i)", E_INVALID_MOVE
, m
, n
);
1596 warnx("%s: %s \"%s\" (round #%i)", loadfile
, E_INVALID_MOVE
, m
, n
);
1599 /* Convert the selected piece to SAN format and validate it. */
1600 static char *board_to_san(GAME
*g
, BOARD b
)
1602 static char str
[MAX_SAN_MOVE_LEN
+ 1], *p
;
1607 snprintf(str
, sizeof(str
), "%c%i%c%i", x_grid_chars
[sp
.col
- 1],
1608 sp
.row
, x_grid_chars
[sp
.destcol
- 1], sp
.destrow
);
1611 piece
= pgn_piece_to_int(b
[ROWTOBOARD(sp
.row
)][COLTOBOARD(sp
.col
)].icon
);
1613 if (piece
== PAWN
&& ((sp
.destrow
== 8 && g
->turn
== WHITE
) ||
1614 (sp
.destrow
== 1 && g
->turn
== BLACK
))) {
1615 promo
= cmessage(PROMOTION_TITLE
, PROMOTION_PROMPT
, PROMOTION_TEXT
);
1617 if (pgn_piece_to_int(promo
) == -1)
1620 p
= str
+ strlen(str
);
1621 *p
++ = toupper(promo
);
1625 memcpy(oldboard
, b
, sizeof(BOARD
));
1627 if ((p
= pgn_a2a4tosan(g
, b
, str
)) == NULL
) {
1628 cmessage(p
, ANYKEY
, "%s", E_A2A4_PARSE
);
1629 memcpy(b
, oldboard
, sizeof(BOARD
));
1633 if (pgn_validate_move(g
, b
, p
)) {
1634 invalid_move(gindex
+ 1, p
);
1635 memcpy(b
, oldboard
, sizeof(BOARD
));
1642 static int move_to_engine(GAME
*g
, BOARD b
)
1646 if ((p
= board_to_san(g
, b
)) == NULL
)
1649 sp
.row
= sp
.col
= sp
.icon
= 0;
1654 SET_FLAG(g
->flags
, GF_MODIFIED
);
1659 send_to_engine(g
, "%s\n", p
);
1663 static void update_clock(int n
, int *h
, int *m
, int *s
)
1666 *m
= (n
% 3600) / 60;
1667 *s
= (n
% 3600) % 60;
1672 void update_status_window(GAME g
)
1676 char tmp
[15], *engine
, *mode
;
1682 struct user_data_s
*d
= g
.data
;
1684 getmaxyx(statusw
, maxy
, maxx
);
1692 if (TEST_FLAG(g
.flags
, GF_DELETE
)) {
1698 if (TEST_FLAG(g
.flags
, GF_PERROR
)) {
1708 if (TEST_FLAG(g
.flags
, GF_MODIFIED
)) {
1723 mvwprintw(statusw
, 2, 1, "%*s %-*s", 7, STATUS_FILE_STR
, w
,
1724 (loadfile
[0]) ? str_etc(loadfile
, w
, 1) : UNAVAILABLE
);
1725 snprintf(buf
, len
, "%i %s %i %s", gindex
+ 1, N_OF_N_STR
, gtotal
,
1727 mvwprintw(statusw
, 3, 1, "%*s %-*s", 7, STATUS_GAME_STR
, w
, buf
);
1731 mode
= MODE_HISTORY_STR
;
1734 mode
= MODE_EDIT_STR
;
1737 mode
= MODE_PLAY_STR
;
1744 snprintf(buf
, len
- 1, "%*s %s%s", 7, STATUS_MODE_STR
, mode
,
1745 (d
&& TEST_FLAG(d
->flags
, CF_ENGINE_LOOP
)) ? " (loop)" : "");
1746 mvwprintw(statusw
, 4, 1, "%-*s", w
, buf
);
1749 switch (d
->status
) {
1750 case ENGINE_THINKING
:
1751 engine
= ENGINE_THINKING_STR
;
1754 engine
= ENGINE_READY_STR
;
1756 case ENGINE_INITIALIZING
:
1757 engine
= ENGINE_INITIALIZING_STR
;
1759 case ENGINE_OFFLINE
:
1760 engine
= ENGINE_OFFLINE_STR
;
1768 engine
= ENGINE_OFFLINE_STR
;
1770 mvwprintw(statusw
, 5, 1, "%*s %-*s", 7, STATUS_ENGINE_STR
, w
, " ");
1771 wattron(statusw
, CP_STATUS_ENGINE
);
1772 mvwaddstr(statusw
, 5, 9, engine
);
1773 wattroff(statusw
, CP_STATUS_ENGINE
);
1775 mvwprintw(statusw
, 6, 1, "%*s %-*s", 7, STATUS_TURN_STR
, w
,
1776 (g
.turn
== WHITE
) ? WHITE_STR
: BLACK_STR
);
1778 strncpy(tmp
, WHITE_STR
, sizeof(tmp
));
1779 tmp
[0] = toupper(tmp
[0]);
1780 update_clock(g
.moveclock
, &h
, &m
, &s
);
1781 snprintf(buf
, len
, "%.2i:%.2i:%.2i", h
, m
, s
);
1782 mvwprintw(statusw
, 7, 1, "%*s: %-*s", 6, tmp
, w
, buf
);
1784 strncpy(tmp
, BLACK_STR
, sizeof(tmp
));
1785 tmp
[0] = toupper(tmp
[0]);
1786 update_clock(g
.moveclock
, &h
, &m
, &s
);
1787 snprintf(buf
, len
, "%.2i:%.2i:%.2i", h
, m
, s
);
1788 mvwprintw(statusw
, 8, 1, "%*s: %-*s", 6, tmp
, w
, buf
);
1791 for (i
= 1; i
< maxx
- 4; i
++)
1792 mvwprintw(statusw
, maxy
- 2, i
, " ");
1795 status
.notify
= strdup(GAME_HELP_PROMPT
);
1797 wattron(statusw
, CP_STATUS_NOTIFY
);
1798 mvwprintw(statusw
, maxy
- 2, CENTERX(maxx
, status
.notify
), "%s",
1800 wattroff(statusw
, CP_STATUS_NOTIFY
);
1803 void update_history_window(GAME g
)
1805 char buf
[HISTORY_WIDTH
- 1];
1808 int t
= history_total(g
.hp
);
1810 n
= (g
.hindex
+ 1) / 2;
1813 total
= (t
+ 1) / 2;
1818 snprintf(buf
, sizeof(buf
), "%u %s %u%s", n
, N_OF_N_STR
, total
,
1819 (movestep
== 1) ? HISTORY_PLY_STEP
: "");
1821 strncpy(buf
, UNAVAILABLE
, sizeof(buf
));
1823 mvwprintw(historyw
, 2, 1, "%*s %-*s", 10, HISTORY_MOVE_STR
,
1824 HISTORY_WIDTH
- 13, buf
);
1826 h
= history_by_n(g
.hp
, g
.hindex
);
1827 snprintf(buf
, sizeof(buf
), "%s", (h
&& h
->move
) ? h
->move
: UNAVAILABLE
);
1830 if (h
&& ((h
->comment
) || h
->nag
[0])) {
1831 strncat(buf
, " (v", sizeof(buf
));
1836 strncat(buf
, (n
) ? ",+" : " (+", sizeof(buf
));
1841 strncat(buf
, (n
) ? ",-" : " (-", sizeof(buf
));
1846 strncat(buf
, ")", sizeof(buf
));
1848 mvwprintw(historyw
, 3, 1, "%s %-*s", HISTORY_MOVE_NEXT_STR
,
1849 HISTORY_WIDTH
- 13, buf
);
1851 h
= history_by_n(g
.hp
, game
[gindex
].hindex
- 1);
1852 snprintf(buf
, sizeof(buf
), "%s", (h
&& h
->move
) ? h
->move
: UNAVAILABLE
);
1855 if (h
&& ((h
->comment
) || h
->nag
[0])) {
1856 strncat(buf
, " (V", sizeof(buf
));
1861 strncat(buf
, (n
) ? ",+" : " (+", sizeof(buf
));
1866 strncat(buf
, (n
) ? ",-" : " (-", sizeof(buf
));
1871 strncat(buf
, ")", sizeof(buf
));
1873 mvwprintw(historyw
, 4, 1, "%s %-*s", HISTORY_MOVE_PREV_STR
,
1874 HISTORY_WIDTH
- 13, buf
);
1877 void update_tag_window(TAG
**t
)
1880 int w
= TAG_WIDTH
- 10;
1882 for (i
= 0; i
< 7; i
++)
1883 mvwprintw(tagw
, (i
+ 2), 1, "%*s: %-*s", 6, t
[i
]->name
, w
, t
[i
]->value
);
1886 void draw_prompt(WINDOW
*win
, int y
, int width
, const char *str
, chtype attr
)
1892 for (i
= 1; i
< width
- 1; i
++)
1893 mvwaddch(win
, y
, i
, ' ');
1895 mvwprintw(win
, y
, CENTERX(width
, str
), "%s", str
);
1896 wattroff(win
, attr
);
1899 void draw_window_title(WINDOW
*win
, const char *title
, int width
, chtype attr
,
1907 for (i
= 1; i
< width
- 1; i
++)
1908 mvwaddch(win
, 1, i
, ' ');
1910 mvwprintw(win
, 1, CENTERX(width
, title
), "%s", title
);
1911 wattroff(win
, attr
);
1914 wattron(win
, battr
);
1915 box(win
, ACS_VLINE
, ACS_HLINE
);
1916 wattroff(win
, battr
);
1925 void update_all(GAME g
)
1927 update_status_window(g
);
1928 update_history_window(g
);
1929 update_tag_window(g
.tag
);
1932 static void game_next_prev(GAME g
, int n
, int count
)
1938 if (gindex
+ count
> gtotal
- 1) {
1940 gindex
= gtotal
- 1;
1948 if (gindex
- count
< 0) {
1952 gindex
= gtotal
- 1;
1959 static void delete_game(int which
)
1965 for (i
= 0; i
< gtotal
; i
++) {
1966 if (i
== which
|| TEST_FLAG(game
[i
].flags
, GF_DELETE
)) {
1971 g
= Realloc(g
, (gi
+ 1) * sizeof(GAME
));
1972 memcpy(&g
[gi
], &game
[i
], sizeof(GAME
));
1973 g
[gi
].tag
= game
[i
].tag
;
1974 g
[gi
].history
= game
[i
].history
;
1975 g
[gi
].hp
= game
[i
].hp
;
1983 if (which
+ 1 >= gtotal
)
1984 gindex
= gtotal
- 1;
1989 gindex
= gtotal
- 1;
1991 game
[gindex
].hp
= game
[gindex
].history
;
1994 static int find_move_exp(GAME g
, const char *str
, int init
, int which
,
2000 static int firstrun
= 1;
2009 if ((ret
= regcomp(&r
, str
, REG_EXTENDED
|REG_NOSUB
)) != 0) {
2010 regerror(ret
, &r
, errbuf
, sizeof(errbuf
));
2011 cmessage(E_REGCOMP_TITLE
, ANYKEY
, "%s", errbuf
);
2018 incr
= (which
== 0) ? -1 : 1;
2020 for (i
= g
.hindex
+ incr
- 1, found
= 0; ; i
+= incr
) {
2021 if (i
== g
.hindex
- 1)
2024 if (i
>= history_total(g
.hp
))
2027 i
= history_total(g
.hp
) - 1;
2030 ret
= regexec(&r
, g
.hp
[i
]->move
, 0, 0, 0);
2033 if (count
== ++found
) {
2038 if (ret
!= REG_NOMATCH
) {
2039 regerror(ret
, &r
, errbuf
, sizeof(errbuf
));
2040 cmessage(E_REGEXEC_TITLE
, ANYKEY
, "%s", errbuf
);
2049 static int toggle_delete_flag(int n
)
2053 TOGGLE_FLAG(game
[n
].flags
, GF_DELETE
);
2055 for (i
= x
= 0; i
< gtotal
; i
++) {
2056 if (TEST_FLAG(game
[i
].flags
, GF_DELETE
))
2061 cmessage(NULL
, ANYKEY
, "%s", E_DELETE_GAME
);
2062 CLEAR_FLAG(game
[n
].flags
, GF_DELETE
);
2069 static void edit_save_tags(GAME
*g
)
2073 if ((t
= edit_tags(*g
, g
->b
, 1)) == NULL
)
2076 pgn_tag_free(g
->tag
);
2078 SET_FLAG(g
->flags
, GF_MODIFIED
);
2079 pgn_sort_tags(g
->tag
);
2082 static int find_game_exp(char *str
, int which
, int count
)
2084 char *nstr
= NULL
, *exp
= NULL
;
2088 char buf
[255], *tmp
;
2091 int incr
= (which
== 0) ? -(1) : 1;
2093 strncpy(buf
, str
, sizeof(buf
));
2096 if (strstr(tmp
, ":") != NULL
) {
2097 nstr
= strsep(&tmp
, ":");
2099 if ((ret
= regcomp(&nexp
, nstr
,
2100 REG_ICASE
|REG_EXTENDED
|REG_NOSUB
)) != 0) {
2101 regerror(ret
, &nexp
, errbuf
, sizeof(errbuf
));
2102 cmessage(E_REGCOMP_TITLE
, ANYKEY
, "%s", errbuf
);
2113 if ((ret
= regcomp(&vexp
, exp
, REG_EXTENDED
|REG_NOSUB
)) != 0) {
2114 regerror(ret
, &vexp
, errbuf
, sizeof(errbuf
));
2115 cmessage(E_REGCOMP_TITLE
, ANYKEY
, "%s", errbuf
);
2122 for (g
= gindex
+ incr
, found
= 0; ; g
+= incr
) {
2133 for (t
= 0; game
[g
].tag
[t
]; t
++) {
2135 if (regexec(&nexp
, game
[g
].tag
[t
]->name
, 0, 0, 0) == 0) {
2136 if (regexec(&vexp
, game
[g
].tag
[t
]->value
, 0, 0, 0) == 0) {
2137 if (count
== ++found
) {
2145 if (regexec(&vexp
, game
[g
].tag
[t
]->value
, 0, 0, 0) == 0) {
2146 if (count
== ++found
) {
2168 * Updates the notification line in the status window then refreshes the
2171 void update_status_notify(GAME g
, char *fmt
, ...)
2174 #ifdef HAVE_VASPRINTF
2181 if (status
.notify
) {
2182 free(status
.notify
);
2183 status
.notify
= NULL
;
2185 if (curses_initialized
)
2186 update_status_window(g
);
2193 #ifdef HAVE_VASPRINTF
2194 vasprintf(&line
, fmt
, ap
);
2196 vsnprintf(line
, sizeof(line
), fmt
, ap
);
2201 free(status
.notify
);
2203 status
.notify
= strdup(line
);
2205 #ifdef HAVE_VASPRINTF
2208 if (curses_initialized
)
2209 update_status_window(g
);
2212 static void switch_side(GAME
*g
)
2214 g
->side
= (g
->side
== WHITE
) ? BLACK
: WHITE
;
2217 int rav_next_prev(GAME
*g
, BOARD b
, int n
)
2221 if (g
->hp
[g
->hindex
]->rav
== NULL
)
2224 g
->rav
= Realloc(g
->rav
, (g
->ravlevel
+ 1) * sizeof(RAV
));
2225 g
->rav
[g
->ravlevel
].hp
= g
->hp
;
2226 g
->rav
[g
->ravlevel
].flags
= g
->flags
;
2227 g
->rav
[g
->ravlevel
].fen
= strdup(pgn_game_to_fen(*g
, b
));
2228 g
->rav
[g
->ravlevel
].hindex
= g
->hindex
;
2229 g
->hp
= g
->hp
[g
->hindex
]->rav
;
2235 if (g
->ravlevel
- 1 < 0)
2240 pgn_init_fen_board(g
, b
, g
->rav
[g
->ravlevel
].fen
);
2241 free(g
->rav
[g
->ravlevel
].fen
);
2242 g
->hp
= g
->rav
[g
->ravlevel
].hp
;
2243 g
->flags
= g
->rav
[g
->ravlevel
].flags
;
2244 g
->hindex
= g
->rav
[g
->ravlevel
].hindex
;
2248 static void draw_window_decor()
2250 move_panel(historyp
, LINES
- HISTORY_HEIGHT
, COLS
- HISTORY_WIDTH
);
2251 move_panel(boardp
, 0, COLS
- BOARD_WIDTH
);
2252 wbkgd(boardw
, CP_BOARD_WINDOW
);
2253 wbkgd(statusw
, CP_STATUS_WINDOW
);
2254 draw_window_title(statusw
, STATUS_WINDOW_TITLE
, STATUS_WIDTH
,
2255 CP_STATUS_TITLE
, CP_STATUS_BORDER
);
2256 wbkgd(tagw
, CP_TAG_WINDOW
);
2257 draw_window_title(tagw
, TAG_WINDOW_TITLE
, TAG_WIDTH
, CP_TAG_TITLE
,
2259 wbkgd(historyw
, CP_HISTORY_WINDOW
);
2260 draw_window_title(historyw
, HISTORY_WINDOW_TITLE
, HISTORY_WIDTH
,
2261 CP_HISTORY_TITLE
, CP_HISTORY_BORDER
);
2264 static void do_window_resize()
2266 if (LINES
< 24 || COLS
< 80)
2269 resizeterm(LINES
, COLS
);
2270 wresize(historyw
, HISTORY_HEIGHT
, HISTORY_WIDTH
);
2271 wresize(statusw
, STATUS_HEIGHT
, STATUS_WIDTH
);
2272 wresize(tagw
, TAG_HEIGHT
, TAG_WIDTH
);
2273 wmove(historyw
, 0, 0);
2274 wclrtobot(historyw
);
2277 wmove(statusw
, 0, 0);
2279 draw_window_decor();
2280 update_all(game
[gindex
]);
2283 static void historymode_keys(int);
2284 static void playmode_keys(int c
)
2286 // More keys in MODE_EDIT share keys with MODE_PLAY than don't.
2287 int editmode
= (game
[gindex
].mode
== MODE_EDIT
) ? 1 : 0;
2291 struct user_data_s
*d
= game
[gindex
].data
;
2298 TOGGLE_FLAG(d
->flags
, CF_ENGINE_LOOP
);
2299 update_all(game
[gindex
]);
2305 if (d
->status
== ENGINE_OFFLINE
)
2310 if ((tmp
= get_input_str_clear(ENGINE_CMD_TITLE
, NULL
)) != NULL
)
2311 send_to_engine(&game
[gindex
], "%s\n", tmp
);
2316 pushkey
= keycount
= 0;
2317 update_status_notify(game
[gindex
], NULL
);
2319 if (!noengine
&& (!d
|| d
->status
== ENGINE_THINKING
)) {
2331 p
= game
[gindex
].b
[ROWTOBOARD(sp
.row
)][COLTOBOARD(sp
.col
)].icon
;
2332 game
[gindex
].b
[ROWTOBOARD(sp
.destrow
)][COLTOBOARD(sp
.destcol
)].icon
= p
;
2333 game
[gindex
].b
[ROWTOBOARD(sp
.row
)][COLTOBOARD(sp
.col
)].icon
= pgn_int_to_piece(game
[gindex
].turn
, OPEN_SQUARE
);
2334 sp
.icon
= sp
.row
= sp
.col
= 0;
2338 if (move_to_engine(&game
[gindex
], game
[gindex
].b
)) {
2339 if (config
.validmoves
)
2340 board_reset_valid_moves(game
[gindex
].b
);
2342 if (TEST_FLAG(game
[gindex
].flags
, GF_GAMEOVER
)) {
2343 CLEAR_FLAG(game
[gindex
].flags
, GF_GAMEOVER
);
2344 SET_FLAG(game
[gindex
].flags
, GF_MODIFIED
);
2350 if (!noengine
&& (!d
|| d
->status
== ENGINE_OFFLINE
) && !editmode
) {
2351 if (start_chess_engine(&game
[gindex
]) < 0) {
2358 wtimeout(boardw
, 70);
2360 if (sp
.icon
|| (!editmode
&& d
&& d
->status
== ENGINE_THINKING
)) {
2365 sp
.icon
= mvwinch(boardw
, ROWTOMATRIX(c_row
),
2366 COLTOMATRIX(c_col
)+1) & A_CHARTEXT
;
2368 if (sp
.icon
== ' ') {
2373 if (!editmode
&& ((islower(sp
.icon
) && game
[gindex
].turn
!= BLACK
)
2374 || (isupper(sp
.icon
) && game
[gindex
].turn
!= WHITE
))) {
2375 message(NULL
, ANYKEY
, "%s", E_SELECT_TURN
);
2383 if (!editmode
&& config
.validmoves
)
2384 board_get_valid_moves(&game
[gindex
], game
[gindex
].b
,
2385 pgn_piece_to_int(sp
.icon
), sp
.row
, sp
.col
, &w
, &x
, &y
, &z
);
2390 send_to_engine(&game
[gindex
], "\nswitch\n");
2391 switch_side(&game
[gindex
]);
2392 update_status_window(game
[gindex
]);
2395 /* FIXME dies reading FIFO sometimes. */
2396 if (!history_total(game
[gindex
].hp
))
2399 history_previous(&game
[gindex
], game
[gindex
].b
, (keycount
) ? keycount
* 2 :
2403 if (status
.engine
== CRAFTY
)
2404 SEND_TO_ENGINE("read %s\n", config
.fifo
);
2406 SEND_TO_ENGINE("\npgnload %s\n", config
.fifo
);
2409 update_history_window(game
[gindex
]);
2412 historymode_keys(c
);
2415 board_details
= (board_details
) ? 0 : 1;
2418 paused
= (paused
) ? 0 : 1;
2421 if (!d
|| d
->status
== ENGINE_OFFLINE
)
2422 start_chess_engine(&game
[gindex
]);
2424 send_to_engine(&game
[gindex
], "go\n");
2431 static void editmode_keys(int c
)
2441 game
[gindex
].b
[ROWTOBOARD(sp
.row
)][COLTOBOARD(sp
.col
)].icon
= pgn_int_to_piece(game
[gindex
].turn
, OPEN_SQUARE
);
2443 game
[gindex
].b
[ROWTOBOARD(c_row
)][COLTOBOARD(c_col
)].icon
= pgn_int_to_piece(game
[gindex
].turn
, OPEN_SQUARE
);
2445 sp
.icon
= sp
.row
= sp
.col
= 0;
2448 pgn_switch_turn(&game
[gindex
]);
2449 switch_side(&game
[gindex
]);
2450 update_all(game
[gindex
]);
2453 castling_state(&game
[gindex
], game
[gindex
].b
, ROWTOBOARD(c_row
),
2455 game
[gindex
].b
[ROWTOBOARD(c_row
)][COLTOBOARD(c_col
)].icon
, 1);
2458 c
= message(GAME_EDIT_TITLE
, GAME_EDIT_PROMPT
, "%s",
2461 if (pgn_piece_to_int(c
) == -1)
2464 game
[gindex
].b
[ROWTOBOARD(c_row
)][COLTOBOARD(c_col
)].icon
= c
;
2467 if (c_row
== 6 || c_row
== 3) {
2468 pgn_reset_enpassant(game
[gindex
].b
);
2469 game
[gindex
].b
[ROWTOBOARD(c_row
)][COLTOBOARD(c_col
)].enpassant
= 1;
2477 static void historymode_keys(int c
)
2481 static char moveexp
[255] = {0};
2485 movestep
= (movestep
== 1) ? 2 : 1;
2486 update_history_window(game
[gindex
]);
2489 history_next(&game
[gindex
], game
[gindex
].b
, (keycount
> 0) ?
2490 config
.jumpcount
* keycount
* movestep
:
2491 config
.jumpcount
* movestep
);
2492 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2493 update_all(game
[gindex
]);
2496 history_previous(&game
[gindex
], game
[gindex
].b
, (keycount
) ?
2497 config
.jumpcount
* keycount
* movestep
:
2498 config
.jumpcount
* movestep
);
2499 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2500 update_all(game
[gindex
]);
2503 history_previous(&game
[gindex
], game
[gindex
].b
, (keycount
) ?
2504 keycount
* movestep
: movestep
);
2505 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2506 update_all(game
[gindex
]);
2509 history_next(&game
[gindex
], game
[gindex
].b
, (keycount
) ?
2510 keycount
* movestep
: movestep
);
2511 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2512 update_all(game
[gindex
]);
2515 n
= game
[gindex
].hindex
;
2517 if (n
&& game
[gindex
].hp
[n
- 1]->move
)
2523 snprintf(buf
, COLS
- 1, "%s \"%s\"", ANNOTATION_EDIT_TITLE
,
2524 game
[gindex
].hp
[n
]->move
);
2526 tmp
= get_input(buf
, game
[gindex
].hp
[n
]->comment
, 0, 0, NAG_PROMPT
,
2527 history_edit_nag
, (void *)game
[gindex
].hp
[n
], CTRL('T'),
2531 if (!tmp
&& (!game
[gindex
].hp
[n
]->comment
||
2532 !*game
[gindex
].hp
[n
]->comment
))
2534 else if (tmp
&& game
[gindex
].hp
[n
]->comment
) {
2535 if (strcmp(tmp
, game
[gindex
].hp
[n
]->comment
) == 0)
2539 len
= (tmp
) ? strlen(tmp
) + 1 : 1;
2540 game
[gindex
].hp
[n
]->comment
= Realloc(game
[gindex
].hp
[n
]->comment
,
2542 strncpy(game
[gindex
].hp
[n
]->comment
, (tmp
) ? tmp
: "", len
);
2543 SET_FLAG(game
[gindex
].flags
, GF_MODIFIED
);
2544 update_all(game
[gindex
]);
2549 if (history_total(game
[gindex
].hp
) < 2)
2554 if (!*moveexp
|| c
== '/') {
2555 if ((tmp
= get_input(FIND_REGEXP
, moveexp
, 1, 1, NULL
, NULL
, NULL
, 0, -1)) == NULL
)
2558 strncpy(moveexp
, tmp
, sizeof(moveexp
));
2562 if ((n
= find_move_exp(game
[gindex
], moveexp
, n
,
2563 (c
== '[') ? 0 : 1, (keycount
) ? keycount
: 1))
2567 game
[gindex
].hindex
= n
;
2568 history_update_board(&game
[gindex
], game
[gindex
].b
, game
[gindex
].hindex
);
2569 update_all(game
[gindex
]);
2570 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2573 view_annotation(*game
[gindex
].hp
[game
[gindex
].hindex
]);
2576 if (game
[gindex
].hindex
- 1 >= 0)
2577 view_annotation(*game
[gindex
].hp
[game
[gindex
].hindex
- 1]);
2581 rav_next_prev(&game
[gindex
], game
[gindex
].b
, (c
== '-') ? 0 : 1);
2582 update_all(game
[gindex
]);
2585 if (history_total(game
[gindex
].hp
) < 2)
2588 /* FIXME field validation
2589 if ((tmp = get_input(GAME_HISTORY_JUMP_TITLE, NULL, 1, 1,
2590 NULL, NULL, NULL, 0, FIELD_TYPE_INTEGER, 1, 0,
2591 game[gindex].htotal)) == NULL)
2596 if ((tmp
= get_input(GAME_HISTORY_JUMP_TITLE
, NULL
, 1, 1,
2597 NULL
, NULL
, NULL
, 0, -1)) == NULL
)
2600 if (!isinteger(tmp
))
2608 if (n
< 0 || n
> (history_total(game
[gindex
].hp
) / 2))
2611 game
[gindex
].hindex
= (n
) ? n
* 2 - 1 : n
* 2;
2612 history_update_board(&game
[gindex
], game
[gindex
].b
,
2613 game
[gindex
].hindex
);
2614 update_all(game
[gindex
]);
2615 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2622 static void cleanup_all_games()
2626 for (i
= 0; i
< gtotal
; i
++) {
2627 struct user_data_s
*d
;
2630 stop_engine(&game
[i
]);
2633 if (d
->fd
[ICS_FD
] > 2)
2634 close(d
->fd
[ICS_FD
]);
2637 game
[i
].data
= NULL
;
2642 // Global and other keys.
2643 static int globalkeys(int c
)
2645 static char gameexp
[255] = {0};
2649 char tfile
[FILENAME_MAX
];
2650 struct user_data_s
*d
= game
[gindex
].data
;
2654 if (game
[gindex
].mode
!= MODE_HISTORY
) {
2655 if (!history_total(game
[gindex
].hp
) ||
2656 (d
&& d
->status
== ENGINE_THINKING
))
2659 game
[gindex
].mode
= MODE_HISTORY
;
2660 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2665 if (TEST_FLAG(game
[gindex
].flags
, GF_BLACK_OPENING
)) {
2666 cmessage(NULL
, ANYKEY
, "%s", E_RESUME_BLACK
);
2670 // FIXME Resuming from previous history could append to a RAV.
2671 if (game
[gindex
].hindex
!= history_total(game
[gindex
].hp
)) {
2673 if ((c
= message(NULL
, YESNO
, "%s",
2674 GAME_RESUME_HISTORY_TEXT
)) != 'y')
2679 if (TEST_FLAG(game
[gindex
].flags
, GF_GAMEOVER
))
2683 if (!noengine
&& (!d
|| d
->status
== ENGINE_OFFLINE
)) {
2684 if (start_chess_engine(&game
[gindex
]) < 0)
2692 oldhistorytotal
= history_total(game
[gindex
].hp
);
2693 game
[gindex
].mode
= MODE_PLAY
;
2694 update_all(game
[gindex
]);
2698 game_next_prev(game
[gindex
], (c
== '>') ? 1 : 0, (keycount
) ?
2707 if (game
[gindex
].mode
!= MODE_EDIT
) {
2708 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2709 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2711 update_all(game
[gindex
]);
2712 update_tag_window(game
[gindex
].tag
);
2714 // Not sure whether to keep these.
2715 case '!': c_row
= 1; return 1;
2716 case '@': c_row
= 2; return 1;
2717 case '#': c_row
= 3; return 1;
2718 case '$': c_row
= 4; return 1;
2719 case '%': c_row
= 5; return 1;
2720 case '^': c_row
= 6; return 1;
2721 case '&': c_row
= 7; return 1;
2722 case '*': c_row
= 8; return 1;
2723 case 'A': c_col
= 1; return 1;
2724 case 'B': c_col
= 2; return 1;
2725 case 'C': c_col
= 3; return 1;
2726 case 'D': c_col
= 4; return 1;
2727 case 'E': c_col
= 5; return 1;
2728 case 'F': c_col
= 6; return 1;
2729 case 'G': c_col
= 7; return 1;
2730 case 'H': c_col
= 8; return 1;
2737 if (!*gameexp
|| c
== '?') {
2738 if ((tmp
= get_input(GAME_FIND_EXPRESSION_TITLE
, gameexp
,
2739 1, 1, GAME_FIND_EXPRESSION_PROMPT
, NULL
,
2740 NULL
, 0, -1)) == NULL
)
2743 strncpy(gameexp
, tmp
, sizeof(gameexp
));
2746 if ((n
= find_game_exp(gameexp
, (c
== '{') ? 0 : 1, (keycount
)
2753 if (history_total(game
[gindex
].hp
))
2754 game
[gindex
].mode
= MODE_HISTORY
;
2756 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2757 update_all(game
[gindex
]);
2758 update_tag_window(game
[gindex
].tag
);
2764 /* FIXME field validation
2765 if ((tmp = get_input(GAME_JUMP_TITLE, NULL, 1, 1, NULL, NULL,
2766 NULL, 0, FIELD_TYPE_INTEGER, 1, 1, gtotal))
2772 if ((tmp
= get_input(GAME_JUMP_TITLE
, NULL
, 1, 1, NULL
,
2773 NULL
, NULL
, 0, -1)) == NULL
)
2776 if (!isinteger(tmp
))
2784 if (--i
> gtotal
- 1 || i
< 0)
2788 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2789 update_cursor(game
[gindex
], game
[gindex
].hindex
);
2790 update_all(game
[gindex
]);
2791 update_tag_window(game
[gindex
].tag
);
2799 if (keycount
&& !delete_count
) {
2802 update_status_notify(game
[gindex
], "%s (delete)",
2807 if (markstart
>= 0 && markend
>= 0) {
2808 if (markstart
> markend
) {
2810 markstart
= markend
;
2814 for (i
= markstart
; i
<= markend
; i
++) {
2815 if (toggle_delete_flag(i
))
2820 if (toggle_delete_flag(gindex
))
2824 markstart
= markend
= -1;
2825 update_status_window(game
[gindex
]);
2829 cmessage(NULL
, ANYKEY
, "%s", E_DELETE_GAME
);
2835 for (i
= n
= 0; i
< gtotal
; i
++) {
2836 if (TEST_FLAG(game
[i
].flags
, GF_DELETE
))
2841 tmp
= GAME_DELETE_GAME_TEXT
;
2844 cmessage(NULL
, ANYKEY
, "%s", E_DELETE_GAME
);
2848 tmp
= GAME_DELETE_ALL_TEXT
;
2851 if (config
.deleteprompt
) {
2852 if ((c
= cmessage(NULL
, YESNO
, "%s", tmp
)) != 'y')
2856 delete_game((!n
) ? gindex
: -1);
2858 if (history_total(game
[gindex
].hp
))
2859 game
[gindex
].mode
= MODE_HISTORY
;
2861 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2862 update_all(game
[gindex
]);
2863 update_tag_window(game
[gindex
].tag
);
2866 edit_save_tags(&game
[gindex
]);
2867 update_all(game
[gindex
]);
2868 update_tag_window(game
[gindex
].tag
);
2871 edit_tags(game
[gindex
], game
[gindex
].b
, 0);
2874 if ((tmp
= get_input(GAME_LOAD_TITLE
, NULL
, 1, 1,
2875 BROWSER_PROMPT
, browse_directory
, NULL
, '\t',
2879 if ((tmp
= word_expand(tmp
)) == NULL
)
2882 if ((fp
= pgn_open(tmp
)) == NULL
) {
2883 cmessage(ERROR
, ANYKEY
, "%s\n%s", tmp
, strerror(errno
));
2890 strncpy(loadfile
, tmp
, sizeof(loadfile
));
2892 if (history_total(game
[gindex
].hp
))
2893 game
[gindex
].mode
= MODE_HISTORY
;
2895 history_update_board(&game
[gindex
], game
[gindex
].b
, history_total(game
[gindex
].hp
));
2896 update_all(game
[gindex
]);
2897 update_tag_window(game
[gindex
].tag
);
2904 n
= message(NULL
, GAME_SAVE_MULTI_PROMPT
, "%s",
2905 GAME_SAVE_MULTI_TEXT
);
2912 update_status_notify(game
[gindex
], "%s", NOTIFY_SAVE_ABORTED
);
2917 if ((tmp
= get_input(GAME_SAVE_TITLE
, loadfile
, 1, 1,
2918 BROWSER_PROMPT
, browse_directory
, NULL
,
2919 '\t', -1)) == NULL
) {
2920 update_status_notify(game
[gindex
], "%s", NOTIFY_SAVE_ABORTED
);
2924 if ((tmp
= word_expand(tmp
)) == NULL
)
2927 if (pgn_is_compressed(tmp
)) {
2928 snprintf(tfile
, sizeof(tfile
), "%s.pgn", tmp
);
2932 if ((p
= strchr(tmp
, '.')) != NULL
) {
2933 if (strcmp(p
, ".pgn") != 0) {
2934 snprintf(tfile
, sizeof(tfile
), "%s.pgn", tmp
);
2939 snprintf(tfile
, sizeof(tfile
), "%s.pgn", tmp
);
2944 if (save_pgn(tmp
, 0, i
)) {
2945 update_status_notify(game
[gindex
], "%s", NOTIFY_SAVE_FAILED
);
2949 update_status_notify(game
[gindex
], "%s", NOTIFY_SAVED
);
2950 update_all(game
[gindex
]);
2956 n
= help(GAME_HELP_INDEX_TITLE
, GAME_HELP_INDEX_PROMPT
,
2961 help(GAME_HELP_HISTORY_TITLE
, ANYKEY
, historyhelp
);
2964 help(GAME_HELP_PLAY_TITLE
, ANYKEY
, playhelp
);
2967 help(GAME_HELP_EDIT_TITLE
, ANYKEY
, edithelp
);
2970 help(GAME_HELP_GAME_TITLE
, ANYKEY
, gamehelp
);
2982 if (cmessage(NULL
, YESNO
, "%s", GAME_NEW_PROMPT
) != 'y')
2988 add_custom_tags(&game
[gindex
].tag
);
2991 cleanup_all_games();
2993 add_custom_tags(&game
[gindex
].tag
);
2994 pgn_init_board(game
[gindex
].b
);
2997 game
[gindex
].mode
= MODE_PLAY
;
2998 c_row
= (game
[gindex
].side
== WHITE
) ? 2 : 7;
3000 update_status_notify(game
[gindex
], NULL
);
3001 update_all(game
[gindex
]);
3002 update_tag_window(game
[gindex
].tag
);
3006 keypad(boardw
, TRUE
);
3010 sp
.icon
= sp
.row
= sp
.col
= 0;
3011 markend
= markstart
= 0;
3015 update_status_notify(game
[gindex
], NULL
);
3018 if (config
.validmoves
)
3019 board_reset_valid_moves(game
[gindex
].b
);
3026 keycount
= keycount
* 10 + n
;
3030 update_status_notify(game
[gindex
], "Repeat %i", keycount
);
3033 if (game
[gindex
].mode
== MODE_HISTORY
)
3048 if (game
[gindex
].mode
== MODE_HISTORY
)
3054 update_status_notify(game
[gindex
], NULL
);
3064 if (game
[gindex
].mode
== MODE_HISTORY
)
3079 if (game
[gindex
].mode
== MODE_HISTORY
)
3094 if (game
[gindex
].mode
!= MODE_EDIT
&& game
[gindex
].mode
!=
3098 // Don't edit a running game (for now).
3099 if (history_total(game
[gindex
].hp
))
3102 if (game
[gindex
].mode
!= MODE_EDIT
) {
3103 pgn_init_fen_board(&game
[gindex
], game
[gindex
].b
, NULL
);
3105 game
[gindex
].mode
= MODE_EDIT
;
3106 update_all(game
[gindex
]);
3111 pgn_add_tag(&game
[gindex
].tag
, "FEN",
3112 pgn_game_to_fen(game
[gindex
], game
[gindex
].b
));
3113 pgn_add_tag(&game
[gindex
].tag
, "SetUp", "1");
3114 pgn_sort_tags(game
[gindex
].tag
);
3115 game
[gindex
].mode
= MODE_PLAY
;
3116 update_all(game
[gindex
]);
3126 message("DEBUG BOARD", ANYKEY
, "%s", debug_board(game
[gindex
].b
));
3139 int error_recover
= 0;
3141 c_row
= 2, c_col
= 5;
3142 gindex
= gtotal
- 1;
3144 if (history_total(game
[gindex
].hp
))
3145 game
[gindex
].mode
= MODE_HISTORY
;
3147 game
[gindex
].mode
= MODE_PLAY
;
3149 if (game
[gindex
].mode
== MODE_HISTORY
) {
3150 history_update_board(&game
[gindex
], game
[gindex
].b
,
3151 history_total(game
[gindex
].hp
));
3152 update_cursor(game
[gindex
], game
[gindex
].hindex
);
3155 update_status_notify(game
[gindex
], "%s", GAME_HELP_PROMPT
);
3157 paused
= 1; //FIXME clock
3159 update_all(game
[gindex
]);
3160 update_tag_window(game
[gindex
].tag
);
3165 char fdbuf
[8192] = {0};
3167 struct timeval tv
= {0, 0};
3169 struct user_data_s
*d
= NULL
;
3174 for (i
= 0; i
< gtotal
; i
++) {
3178 if (d
->fd
[ENGINE_IN_FD
] > 2) {
3179 if (d
->fd
[ENGINE_IN_FD
] > n
)
3180 n
= d
->fd
[ENGINE_IN_FD
];
3182 FD_SET(d
->fd
[ENGINE_IN_FD
], &rfds
);
3185 if (d
->fd
[ICS_FD
] > 2) {
3186 if (d
->fd
[ICS_FD
] > n
)
3189 FD_SET(d
->fd
[ICS_FD
], &rfds
);
3190 FD_SET(d
->fd
[ICS_FD
], &wfds
);
3196 if ((n
= select(n
+ 1, &rfds
, &wfds
, NULL
, &tv
)) > 0) {
3197 for (i
= 0; i
< gtotal
; i
++) {
3201 if (FD_ISSET(d
->fd
[ENGINE_IN_FD
], &rfds
)) {
3202 len
= read(d
->fd
[ENGINE_IN_FD
], fdbuf
,
3206 if (errno
!= EAGAIN
) {
3207 cmessage(ERROR
, ANYKEY
, "Attempt #%i. read(): %s",
3208 ++error_recover
, strerror(errno
));
3214 parse_engine_output(&game
[gindex
], fdbuf
);
3215 update_all(game
[gindex
]);
3224 cmessage(ERROR
, ANYKEY
, "select(): %s", strerror(errno
));
3232 draw_board(&game
[gindex
], board_details
);
3233 wmove(boardw
, ROWTOMATRIX(c_row
), COLTOMATRIX(c_col
));
3243 if ((c
= wgetch(boardw
)) == ERR
)
3247 if (!keycount
&& status
.notify
)
3248 update_status_notify(game
[gindex
], NULL
);
3251 if ((n
= globalkeys(c
)) == 1) {
3258 switch (game
[gindex
].mode
) {
3266 historymode_keys(c
);
3276 void usage(const char *pn
, int ret
)
3278 fprintf((ret
) ? stderr
: stdout
, "%s",
3279 "Usage: cboard [-hvNE] [-VtRS] [-p <file>]\n"
3280 " -p Load PGN file.\n"
3281 " -V Validate a game file.\n"
3282 " -S Validate and output a PGN formatted game.\n"
3283 " -R Like -S but write a reduced PGN formatted game.\n"
3284 " -t Also write custom PGN tags from config file.\n"
3285 " -N Don't enable the chess engine (two human players).\n"
3286 " -E Stop processing on file parsing error (overrides config).\n"
3287 " -v Version information.\n"
3288 " -h This help text.\n");
3295 cleanup_all_games();
3298 del_panel(historyp
);
3308 void catch_signal(int which
)
3313 if (which
== SIGPIPE
&& quit
)
3316 if (which
== SIGPIPE
)
3317 cmessage(NULL
, ANYKEY
, "%s", E_BROKEN_PIPE
);
3327 keypad(boardw
, TRUE
);
3337 static void set_defaults()
3340 set_config_defaults();
3343 int main(int argc
, char *argv
[])
3347 char buf
[FILENAME_MAX
];
3348 char datadir
[FILENAME_MAX
];
3349 int ret
= EXIT_SUCCESS
;
3350 int validate_only
= 0, validate_and_write
= 0, reduced
= 0;
3351 int write_custom_tags
= 0;
3354 if ((config
.pwd
= getpwuid(getuid())) == NULL
)
3355 err(EXIT_FAILURE
, "getpwuid()");
3357 snprintf(datadir
, sizeof(datadir
), "%s/.cboard", config
.pwd
->pw_dir
);
3358 snprintf(buf
, sizeof(buf
), "%s/cc.data", datadir
);
3359 config
.ccfile
= strdup(buf
);
3360 snprintf(buf
, sizeof(buf
), "%s/nag.data", datadir
);
3361 config
.nagfile
= strdup(buf
);
3362 snprintf(buf
, sizeof(buf
), "%s/agony.data", datadir
);
3363 config
.agonyfile
= strdup(buf
);
3364 snprintf(buf
, sizeof(buf
), "%s/config", datadir
);
3365 config
.configfile
= strdup(buf
);
3366 snprintf(buf
, sizeof(buf
), "%s/fifo", datadir
);
3367 config
.fifo
= strdup(buf
);
3369 if (stat(datadir
, &st
) == -1) {
3370 if (errno
== ENOENT
) {
3371 if (mkdir(datadir
, 0755) == -1)
3372 err(EXIT_FAILURE
, "%s", datadir
);
3375 err(EXIT_FAILURE
, "%s", datadir
);
3380 if (!S_ISDIR(st
.st_mode
))
3381 errx(EXIT_FAILURE
, "%s: %s", datadir
, E_NOTADIR
);
3383 if (access(config
.fifo
, R_OK
) == -1 && errno
== ENOENT
) {
3384 if (mkfifo(config
.fifo
, 0600) == -1)
3385 err(EXIT_FAILURE
, "%s", config
.fifo
);
3390 while ((opt
= getopt(argc
, argv
, "ENVtSRhp:v")) != -1) {
3393 write_custom_tags
= 1;
3396 config
.stoponerror
= 1;
3404 validate_and_write
= 1;
3409 printf("%s (%s)\n%s\n", PACKAGE_STRING
, curses_version(),
3413 filetype
= PGN_FILE
;
3414 strncpy(loadfile
, optarg
, sizeof(loadfile
));
3418 usage(argv
[0], EXIT_SUCCESS
);
3422 if ((validate_only
|| validate_and_write
) && !*loadfile
)
3423 usage(argv
[0], EXIT_FAILURE
);
3425 if (access(config
.configfile
, R_OK
) == 0)
3426 parse_rcfile(config
.configfile
);
3428 signal(SIGPIPE
, catch_signal
);
3429 signal(SIGCONT
, catch_signal
);
3430 signal(SIGSTOP
, catch_signal
);
3431 signal(SIGINT
, catch_signal
);
3437 if ((fp
= pgn_open(loadfile
)) == NULL
)
3438 err(EXIT_FAILURE
, "%s", loadfile
);
3440 ret
= pgn_parse(fp
);
3443 //ret = parse_fen_file(loadfile);
3445 case EPD_FILE
: // Not implemented.
3448 // No file specified. Empty game.
3449 ret
= pgn_parse(NULL
);
3450 add_custom_tags(&game
[gindex
].tag
);
3454 if (validate_only
|| validate_and_write
) {
3455 if (validate_and_write
) {
3458 for (i
= 0; i
< gtotal
; i
++) {
3459 if (write_custom_tags
)
3460 add_custom_tags(&game
[i
].tag
);
3462 pgn_write(stdout
, game
[i
]);
3472 if (initscr() == NULL
)
3473 errx(EXIT_FAILURE
, "%s", E_INITCURSES
);
3475 curses_initialized
= 1;
3477 if (LINES
< 24 || COLS
< 80) {
3479 errx(EXIT_FAILURE
, "Need at least an 80x24 terminal.");
3482 if (has_colors() == TRUE
&& start_color() == OK
)
3485 boardw
= newwin(BOARD_HEIGHT
, BOARD_WIDTH
, 0, COLS
- BOARD_WIDTH
);
3486 boardp
= new_panel(boardw
);
3487 historyw
= newwin(HISTORY_HEIGHT
, HISTORY_WIDTH
, LINES
- HISTORY_HEIGHT
,
3488 COLS
- HISTORY_WIDTH
);
3489 historyp
= new_panel(historyw
);
3490 statusw
= newwin(STATUS_HEIGHT
, STATUS_WIDTH
, LINES
- STATUS_HEIGHT
, 0);
3491 statusp
= new_panel(statusw
);
3492 tagw
= newwin(TAG_HEIGHT
, TAG_WIDTH
, 0, 0);
3493 tagp
= new_panel(tagw
);
3494 keypad(boardw
, TRUE
);
3495 // leaveok(boardw, TRUE);
3496 leaveok(tagw
, TRUE
);
3497 leaveok(statusw
, TRUE
);
3498 leaveok(historyw
, TRUE
);
3502 draw_window_decor();