14c8945a0SNathan Whitehorn /*
2a96ef450SBaptiste Daroussin * $Id: util.c,v 1.300 2021/01/17 22:10:56 tom Exp $
34c8945a0SNathan Whitehorn *
44c8945a0SNathan Whitehorn * util.c -- miscellaneous utilities for dialog
54c8945a0SNathan Whitehorn *
6a96ef450SBaptiste Daroussin * Copyright 2000-2020,2021 Thomas E. Dickey
74c8945a0SNathan Whitehorn *
84c8945a0SNathan Whitehorn * This program is free software; you can redistribute it and/or modify
94c8945a0SNathan Whitehorn * it under the terms of the GNU Lesser General Public License, version 2.1
104c8945a0SNathan Whitehorn * as published by the Free Software Foundation.
114c8945a0SNathan Whitehorn *
124c8945a0SNathan Whitehorn * This program is distributed in the hope that it will be useful, but
134c8945a0SNathan Whitehorn * WITHOUT ANY WARRANTY; without even the implied warranty of
144c8945a0SNathan Whitehorn * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
154c8945a0SNathan Whitehorn * Lesser General Public License for more details.
164c8945a0SNathan Whitehorn *
174c8945a0SNathan Whitehorn * You should have received a copy of the GNU Lesser General Public
184c8945a0SNathan Whitehorn * License along with this program; if not, write to
194c8945a0SNathan Whitehorn * Free Software Foundation, Inc.
204c8945a0SNathan Whitehorn * 51 Franklin St., Fifth Floor
214c8945a0SNathan Whitehorn * Boston, MA 02110, USA.
224c8945a0SNathan Whitehorn *
234c8945a0SNathan Whitehorn * An earlier version of this program lists as authors
244c8945a0SNathan Whitehorn * Savio Lam (lam836@cs.cuhk.hk)
254c8945a0SNathan Whitehorn */
264c8945a0SNathan Whitehorn
274c8945a0SNathan Whitehorn #include <dialog.h>
284c8945a0SNathan Whitehorn #include <dlg_keys.h>
29a96ef450SBaptiste Daroussin #include <dlg_internals.h>
30a96ef450SBaptiste Daroussin
31a96ef450SBaptiste Daroussin #include <sys/time.h>
324c8945a0SNathan Whitehorn
332a3e3873SBaptiste Daroussin #ifdef HAVE_SETLOCALE
342a3e3873SBaptiste Daroussin #include <locale.h>
352a3e3873SBaptiste Daroussin #endif
362a3e3873SBaptiste Daroussin
372a3e3873SBaptiste Daroussin #ifdef NEED_WCHAR_H
382a3e3873SBaptiste Daroussin #include <wchar.h>
392a3e3873SBaptiste Daroussin #endif
402a3e3873SBaptiste Daroussin
41a96ef450SBaptiste Daroussin #ifdef HAVE_SYS_PARAM_H
42*95da5e13SBaptiste Daroussin #undef MIN
43*95da5e13SBaptiste Daroussin #undef MAX
44a96ef450SBaptiste Daroussin #include <sys/param.h>
45a96ef450SBaptiste Daroussin #endif
46a96ef450SBaptiste Daroussin
47a96ef450SBaptiste Daroussin #if defined(NCURSES_VERSION)
48a96ef450SBaptiste Daroussin #define CAN_KEEP_TITE 1
49a96ef450SBaptiste Daroussin #elif defined(__NetBSD_Version__) && (__NetBSD_Version__ >= 800000000)
50a96ef450SBaptiste Daroussin #define CAN_KEEP_TITE 1
51a96ef450SBaptiste Daroussin #else
52a96ef450SBaptiste Daroussin #define CAN_KEEP_TITE 0
53a96ef450SBaptiste Daroussin #endif
54a96ef450SBaptiste Daroussin
55a96ef450SBaptiste Daroussin #if CAN_KEEP_TITE
56a96ef450SBaptiste Daroussin #if defined(NCURSES_VERSION) && defined(HAVE_NCURSESW_TERM_H)
574c8945a0SNathan Whitehorn #include <ncursesw/term.h>
58a96ef450SBaptiste Daroussin #elif defined(NCURSES_VERSION) && defined(HAVE_NCURSES_TERM_H)
594c8945a0SNathan Whitehorn #include <ncurses/term.h>
604c8945a0SNathan Whitehorn #else
614c8945a0SNathan Whitehorn #include <term.h>
624c8945a0SNathan Whitehorn #endif
634c8945a0SNathan Whitehorn #endif
644c8945a0SNathan Whitehorn
65682c9e0fSNathan Whitehorn #if defined(HAVE_WCHGAT)
66682c9e0fSNathan Whitehorn # if defined(NCURSES_VERSION_PATCH)
67682c9e0fSNathan Whitehorn # if NCURSES_VERSION_PATCH >= 20060715
68682c9e0fSNathan Whitehorn # define USE_WCHGAT 1
69682c9e0fSNathan Whitehorn # else
70682c9e0fSNathan Whitehorn # define USE_WCHGAT 0
71682c9e0fSNathan Whitehorn # endif
72682c9e0fSNathan Whitehorn # else
73682c9e0fSNathan Whitehorn # define USE_WCHGAT 1
74682c9e0fSNathan Whitehorn # endif
75682c9e0fSNathan Whitehorn #else
76682c9e0fSNathan Whitehorn # define USE_WCHGAT 0
77682c9e0fSNathan Whitehorn #endif
78682c9e0fSNathan Whitehorn
794c8945a0SNathan Whitehorn /* globals */
804c8945a0SNathan Whitehorn DIALOG_STATE dialog_state;
814c8945a0SNathan Whitehorn DIALOG_VARS dialog_vars;
824c8945a0SNathan Whitehorn
832a3e3873SBaptiste Daroussin #if !(defined(HAVE_WGETPARENT) && defined(HAVE_WINDOW__PARENT))
842a3e3873SBaptiste Daroussin #define NEED_WGETPARENT 1
852a3e3873SBaptiste Daroussin #else
862a3e3873SBaptiste Daroussin #undef NEED_WGETPARENT
872a3e3873SBaptiste Daroussin #endif
882a3e3873SBaptiste Daroussin
894c8945a0SNathan Whitehorn #define concat(a,b) a##b
904c8945a0SNathan Whitehorn
914c8945a0SNathan Whitehorn #ifdef HAVE_RC_FILE
924c8945a0SNathan Whitehorn #define RC_DATA(name,comment) , #name "_color", comment " color"
934c8945a0SNathan Whitehorn #else
944c8945a0SNathan Whitehorn #define RC_DATA(name,comment) /*nothing */
954c8945a0SNathan Whitehorn #endif
964c8945a0SNathan Whitehorn
974c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
984c8945a0SNathan Whitehorn #include <dlg_colors.h>
99a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
100a96ef450SBaptiste Daroussin #define COLOR_DATA(upr) , \
101a96ef450SBaptiste Daroussin concat(DLGC_FG_,upr), \
102a96ef450SBaptiste Daroussin concat(DLGC_BG_,upr), \
103a96ef450SBaptiste Daroussin concat(DLGC_HL_,upr), \
104a96ef450SBaptiste Daroussin concat(DLGC_UL_,upr), \
105a96ef450SBaptiste Daroussin concat(DLGC_RV_,upr)
106a96ef450SBaptiste Daroussin #else /* HAVE_RC_FILE2 */
1074c8945a0SNathan Whitehorn #define COLOR_DATA(upr) , \
1084c8945a0SNathan Whitehorn concat(DLGC_FG_,upr), \
1094c8945a0SNathan Whitehorn concat(DLGC_BG_,upr), \
1104c8945a0SNathan Whitehorn concat(DLGC_HL_,upr)
111a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
112a96ef450SBaptiste Daroussin #else /* HAVE_COLOR */
1134c8945a0SNathan Whitehorn #define COLOR_DATA(upr) /*nothing */
114a96ef450SBaptiste Daroussin #endif /* HAVE_COLOR */
1154c8945a0SNathan Whitehorn
116682c9e0fSNathan Whitehorn #define UseShadow(dw) ((dw) != 0 && (dw)->normal != 0 && (dw)->shadow != 0)
117682c9e0fSNathan Whitehorn
1184c8945a0SNathan Whitehorn /*
1194c8945a0SNathan Whitehorn * Table of color and attribute values, default is for mono display.
1202a3e3873SBaptiste Daroussin * The order matches the DIALOG_ATR() values.
1214c8945a0SNathan Whitehorn */
122a96ef450SBaptiste Daroussin #define DATA(atr,upr,lwr,cmt) { atr COLOR_DATA(upr) RC_DATA(lwr,cmt) }
1234c8945a0SNathan Whitehorn /* *INDENT-OFF* */
1244c8945a0SNathan Whitehorn DIALOG_COLORS dlg_color_table[] =
1254c8945a0SNathan Whitehorn {
1264c8945a0SNathan Whitehorn DATA(A_NORMAL, SCREEN, screen, "Screen"),
1274c8945a0SNathan Whitehorn DATA(A_NORMAL, SHADOW, shadow, "Shadow"),
1284c8945a0SNathan Whitehorn DATA(A_REVERSE, DIALOG, dialog, "Dialog box"),
1294c8945a0SNathan Whitehorn DATA(A_REVERSE, TITLE, title, "Dialog box title"),
1304c8945a0SNathan Whitehorn DATA(A_REVERSE, BORDER, border, "Dialog box border"),
1314c8945a0SNathan Whitehorn DATA(A_BOLD, BUTTON_ACTIVE, button_active, "Active button"),
1324c8945a0SNathan Whitehorn DATA(A_DIM, BUTTON_INACTIVE, button_inactive, "Inactive button"),
1334c8945a0SNathan Whitehorn DATA(A_UNDERLINE, BUTTON_KEY_ACTIVE, button_key_active, "Active button key"),
1344c8945a0SNathan Whitehorn DATA(A_UNDERLINE, BUTTON_KEY_INACTIVE, button_key_inactive, "Inactive button key"),
1354c8945a0SNathan Whitehorn DATA(A_NORMAL, BUTTON_LABEL_ACTIVE, button_label_active, "Active button label"),
1364c8945a0SNathan Whitehorn DATA(A_NORMAL, BUTTON_LABEL_INACTIVE, button_label_inactive, "Inactive button label"),
1374c8945a0SNathan Whitehorn DATA(A_REVERSE, INPUTBOX, inputbox, "Input box"),
1384c8945a0SNathan Whitehorn DATA(A_REVERSE, INPUTBOX_BORDER, inputbox_border, "Input box border"),
1394c8945a0SNathan Whitehorn DATA(A_REVERSE, SEARCHBOX, searchbox, "Search box"),
1404c8945a0SNathan Whitehorn DATA(A_REVERSE, SEARCHBOX_TITLE, searchbox_title, "Search box title"),
1414c8945a0SNathan Whitehorn DATA(A_REVERSE, SEARCHBOX_BORDER, searchbox_border, "Search box border"),
1424c8945a0SNathan Whitehorn DATA(A_REVERSE, POSITION_INDICATOR, position_indicator, "File position indicator"),
1434c8945a0SNathan Whitehorn DATA(A_REVERSE, MENUBOX, menubox, "Menu box"),
1444c8945a0SNathan Whitehorn DATA(A_REVERSE, MENUBOX_BORDER, menubox_border, "Menu box border"),
1454c8945a0SNathan Whitehorn DATA(A_REVERSE, ITEM, item, "Item"),
1464c8945a0SNathan Whitehorn DATA(A_NORMAL, ITEM_SELECTED, item_selected, "Selected item"),
1474c8945a0SNathan Whitehorn DATA(A_REVERSE, TAG, tag, "Tag"),
1484c8945a0SNathan Whitehorn DATA(A_REVERSE, TAG_SELECTED, tag_selected, "Selected tag"),
1494c8945a0SNathan Whitehorn DATA(A_NORMAL, TAG_KEY, tag_key, "Tag key"),
1504c8945a0SNathan Whitehorn DATA(A_BOLD, TAG_KEY_SELECTED, tag_key_selected, "Selected tag key"),
1514c8945a0SNathan Whitehorn DATA(A_REVERSE, CHECK, check, "Check box"),
1524c8945a0SNathan Whitehorn DATA(A_REVERSE, CHECK_SELECTED, check_selected, "Selected check box"),
1534c8945a0SNathan Whitehorn DATA(A_REVERSE, UARROW, uarrow, "Up arrow"),
1544c8945a0SNathan Whitehorn DATA(A_REVERSE, DARROW, darrow, "Down arrow"),
1554c8945a0SNathan Whitehorn DATA(A_NORMAL, ITEMHELP, itemhelp, "Item help-text"),
1564c8945a0SNathan Whitehorn DATA(A_BOLD, FORM_ACTIVE_TEXT, form_active_text, "Active form text"),
1574c8945a0SNathan Whitehorn DATA(A_REVERSE, FORM_TEXT, form_text, "Form text"),
1587a1c0d96SNathan Whitehorn DATA(A_NORMAL, FORM_ITEM_READONLY, form_item_readonly, "Readonly form item"),
1592a3e3873SBaptiste Daroussin DATA(A_REVERSE, GAUGE, gauge, "Dialog box gauge"),
1602a3e3873SBaptiste Daroussin DATA(A_REVERSE, BORDER2, border2, "Dialog box border2"),
1612a3e3873SBaptiste Daroussin DATA(A_REVERSE, INPUTBOX_BORDER2, inputbox_border2, "Input box border2"),
1622a3e3873SBaptiste Daroussin DATA(A_REVERSE, SEARCHBOX_BORDER2, searchbox_border2, "Search box border2"),
1632a3e3873SBaptiste Daroussin DATA(A_REVERSE, MENUBOX_BORDER2, menubox_border2, "Menu box border2")
1644c8945a0SNathan Whitehorn };
165a96ef450SBaptiste Daroussin #undef DATA
1664c8945a0SNathan Whitehorn /* *INDENT-ON* */
1674c8945a0SNathan Whitehorn
1684c8945a0SNathan Whitehorn /*
1692a3e3873SBaptiste Daroussin * Maintain a list of subwindows so that we can delete them to cleanup.
1702a3e3873SBaptiste Daroussin * More important, this provides a fallback when wgetparent() is not available.
1712a3e3873SBaptiste Daroussin */
1722a3e3873SBaptiste Daroussin static void
add_subwindow(WINDOW * parent,WINDOW * child)1732a3e3873SBaptiste Daroussin add_subwindow(WINDOW *parent, WINDOW *child)
1742a3e3873SBaptiste Daroussin {
1752a3e3873SBaptiste Daroussin DIALOG_WINDOWS *p = dlg_calloc(DIALOG_WINDOWS, 1);
1762a3e3873SBaptiste Daroussin
1772a3e3873SBaptiste Daroussin if (p != 0) {
1782a3e3873SBaptiste Daroussin p->normal = parent;
1792a3e3873SBaptiste Daroussin p->shadow = child;
180a96ef450SBaptiste Daroussin p->getc_timeout = WTIMEOUT_OFF;
1812a3e3873SBaptiste Daroussin p->next = dialog_state.all_subwindows;
1822a3e3873SBaptiste Daroussin dialog_state.all_subwindows = p;
1832a3e3873SBaptiste Daroussin }
1842a3e3873SBaptiste Daroussin }
1852a3e3873SBaptiste Daroussin
1862a3e3873SBaptiste Daroussin static void
del_subwindows(WINDOW * parent)1872a3e3873SBaptiste Daroussin del_subwindows(WINDOW *parent)
1882a3e3873SBaptiste Daroussin {
1892a3e3873SBaptiste Daroussin DIALOG_WINDOWS *p = dialog_state.all_subwindows;
1902a3e3873SBaptiste Daroussin DIALOG_WINDOWS *q = 0;
1912a3e3873SBaptiste Daroussin DIALOG_WINDOWS *r;
1922a3e3873SBaptiste Daroussin
1932a3e3873SBaptiste Daroussin while (p != 0) {
1942a3e3873SBaptiste Daroussin if (p->normal == parent) {
1952a3e3873SBaptiste Daroussin delwin(p->shadow);
1962a3e3873SBaptiste Daroussin r = p->next;
1972a3e3873SBaptiste Daroussin if (q == 0) {
1982a3e3873SBaptiste Daroussin dialog_state.all_subwindows = r;
1992a3e3873SBaptiste Daroussin } else {
2002a3e3873SBaptiste Daroussin q->next = r;
2012a3e3873SBaptiste Daroussin }
2022a3e3873SBaptiste Daroussin free(p);
2032a3e3873SBaptiste Daroussin p = r;
2042a3e3873SBaptiste Daroussin } else {
2052a3e3873SBaptiste Daroussin q = p;
2062a3e3873SBaptiste Daroussin p = p->next;
2072a3e3873SBaptiste Daroussin }
2082a3e3873SBaptiste Daroussin }
2092a3e3873SBaptiste Daroussin }
2102a3e3873SBaptiste Daroussin
2112a3e3873SBaptiste Daroussin /*
2124c8945a0SNathan Whitehorn * Display background title if it exists ...
2134c8945a0SNathan Whitehorn */
2144c8945a0SNathan Whitehorn void
dlg_put_backtitle(void)2154c8945a0SNathan Whitehorn dlg_put_backtitle(void)
2164c8945a0SNathan Whitehorn {
2174c8945a0SNathan Whitehorn
2184c8945a0SNathan Whitehorn if (dialog_vars.backtitle != NULL) {
2194c8945a0SNathan Whitehorn chtype attr = A_NORMAL;
2204c8945a0SNathan Whitehorn int backwidth = dlg_count_columns(dialog_vars.backtitle);
221a96ef450SBaptiste Daroussin int i;
2224c8945a0SNathan Whitehorn
223f4f33ea0SBaptiste Daroussin dlg_attrset(stdscr, screen_attr);
2244c8945a0SNathan Whitehorn (void) wmove(stdscr, 0, 1);
2254c8945a0SNathan Whitehorn dlg_print_text(stdscr, dialog_vars.backtitle, COLS - 2, &attr);
2264c8945a0SNathan Whitehorn for (i = 0; i < COLS - backwidth; i++)
2274c8945a0SNathan Whitehorn (void) waddch(stdscr, ' ');
2284c8945a0SNathan Whitehorn (void) wmove(stdscr, 1, 1);
2294c8945a0SNathan Whitehorn for (i = 0; i < COLS - 2; i++)
2304c8945a0SNathan Whitehorn (void) waddch(stdscr, dlg_boxchar(ACS_HLINE));
2314c8945a0SNathan Whitehorn }
2324c8945a0SNathan Whitehorn
2334c8945a0SNathan Whitehorn (void) wnoutrefresh(stdscr);
2344c8945a0SNathan Whitehorn }
2354c8945a0SNathan Whitehorn
2364c8945a0SNathan Whitehorn /*
2374c8945a0SNathan Whitehorn * Set window to attribute 'attr'. There are more efficient ways to do this,
2384c8945a0SNathan Whitehorn * but will not work on older/buggy ncurses versions.
2394c8945a0SNathan Whitehorn */
2404c8945a0SNathan Whitehorn void
dlg_attr_clear(WINDOW * win,int height,int width,chtype attr)2414c8945a0SNathan Whitehorn dlg_attr_clear(WINDOW *win, int height, int width, chtype attr)
2424c8945a0SNathan Whitehorn {
2434c8945a0SNathan Whitehorn int i, j;
2444c8945a0SNathan Whitehorn
245f4f33ea0SBaptiste Daroussin dlg_attrset(win, attr);
2464c8945a0SNathan Whitehorn for (i = 0; i < height; i++) {
2474c8945a0SNathan Whitehorn (void) wmove(win, i, 0);
2484c8945a0SNathan Whitehorn for (j = 0; j < width; j++)
2494c8945a0SNathan Whitehorn (void) waddch(win, ' ');
2504c8945a0SNathan Whitehorn }
2514c8945a0SNathan Whitehorn (void) touchwin(win);
2524c8945a0SNathan Whitehorn }
2534c8945a0SNathan Whitehorn
2544c8945a0SNathan Whitehorn void
dlg_clear(void)2554c8945a0SNathan Whitehorn dlg_clear(void)
2564c8945a0SNathan Whitehorn {
2574c8945a0SNathan Whitehorn dlg_attr_clear(stdscr, LINES, COLS, screen_attr);
2584c8945a0SNathan Whitehorn }
2594c8945a0SNathan Whitehorn
260a96ef450SBaptiste Daroussin #ifdef KEY_RESIZE
261a96ef450SBaptiste Daroussin void
_dlg_resize_cleanup(WINDOW * w)262a96ef450SBaptiste Daroussin _dlg_resize_cleanup(WINDOW *w)
263a96ef450SBaptiste Daroussin {
264a96ef450SBaptiste Daroussin dlg_clear();
265a96ef450SBaptiste Daroussin dlg_put_backtitle();
266a96ef450SBaptiste Daroussin dlg_del_window(w);
267a96ef450SBaptiste Daroussin dlg_mouse_free_regions();
268a96ef450SBaptiste Daroussin }
269a96ef450SBaptiste Daroussin #endif /* KEY_RESIZE */
270a96ef450SBaptiste Daroussin
2714c8945a0SNathan Whitehorn #define isprivate(s) ((s) != 0 && strstr(s, "\033[?") != 0)
2724c8945a0SNathan Whitehorn
2734c8945a0SNathan Whitehorn #define TTY_DEVICE "/dev/tty"
2744c8945a0SNathan Whitehorn
2754c8945a0SNathan Whitehorn /*
2764c8945a0SNathan Whitehorn * If $DIALOG_TTY exists, allow the program to try to open the terminal
2774c8945a0SNathan Whitehorn * directly when stdout is redirected. By default we require the "--stdout"
2784c8945a0SNathan Whitehorn * option to be given, but some scripts were written making use of the
2794c8945a0SNathan Whitehorn * behavior of dialog which tried opening the terminal anyway.
2804c8945a0SNathan Whitehorn */
281a96ef450SBaptiste Daroussin #define dialog_tty() (dlg_getenv_num("DIALOG_TTY", (int *)0) > 0)
2824c8945a0SNathan Whitehorn
2834c8945a0SNathan Whitehorn /*
2844c8945a0SNathan Whitehorn * Open the terminal directly. If one of stdin, stdout or stderr really points
2854c8945a0SNathan Whitehorn * to a tty, use it. Otherwise give up and open /dev/tty.
2864c8945a0SNathan Whitehorn */
2874c8945a0SNathan Whitehorn static int
open_terminal(char ** result,int mode)2884c8945a0SNathan Whitehorn open_terminal(char **result, int mode)
2894c8945a0SNathan Whitehorn {
2904c8945a0SNathan Whitehorn const char *device = TTY_DEVICE;
2914c8945a0SNathan Whitehorn if (!isatty(fileno(stderr))
2924c8945a0SNathan Whitehorn || (device = ttyname(fileno(stderr))) == 0) {
2934c8945a0SNathan Whitehorn if (!isatty(fileno(stdout))
2944c8945a0SNathan Whitehorn || (device = ttyname(fileno(stdout))) == 0) {
2954c8945a0SNathan Whitehorn if (!isatty(fileno(stdin))
2964c8945a0SNathan Whitehorn || (device = ttyname(fileno(stdin))) == 0) {
2974c8945a0SNathan Whitehorn device = TTY_DEVICE;
2984c8945a0SNathan Whitehorn }
2994c8945a0SNathan Whitehorn }
3004c8945a0SNathan Whitehorn }
3014c8945a0SNathan Whitehorn *result = dlg_strclone(device);
3024c8945a0SNathan Whitehorn return open(device, mode);
3034c8945a0SNathan Whitehorn }
3044c8945a0SNathan Whitehorn
305a96ef450SBaptiste Daroussin #if CAN_KEEP_TITE
3062a3e3873SBaptiste Daroussin static int
my_putc(int ch)3072a3e3873SBaptiste Daroussin my_putc(int ch)
3082a3e3873SBaptiste Daroussin {
3092a3e3873SBaptiste Daroussin char buffer[2];
3102a3e3873SBaptiste Daroussin int fd = fileno(dialog_state.screen_output);
3112a3e3873SBaptiste Daroussin
3122a3e3873SBaptiste Daroussin buffer[0] = (char) ch;
3132a3e3873SBaptiste Daroussin return (int) write(fd, buffer, (size_t) 1);
3142a3e3873SBaptiste Daroussin }
3152a3e3873SBaptiste Daroussin #endif
3162a3e3873SBaptiste Daroussin
3174c8945a0SNathan Whitehorn /*
3184c8945a0SNathan Whitehorn * Do some initialization for dialog.
3194c8945a0SNathan Whitehorn *
3204c8945a0SNathan Whitehorn * 'input' is the real tty input of dialog. Usually it is stdin, but if
3214c8945a0SNathan Whitehorn * --input-fd option is used, it may be anything.
3224c8945a0SNathan Whitehorn *
3234c8945a0SNathan Whitehorn * 'output' is where dialog will send its result. Usually it is stderr, but
3244c8945a0SNathan Whitehorn * if --stdout or --output-fd is used, it may be anything. We are concerned
3254c8945a0SNathan Whitehorn * mainly with the case where it happens to be the same as stdout.
3264c8945a0SNathan Whitehorn */
3274c8945a0SNathan Whitehorn void
init_dialog(FILE * input,FILE * output)3284c8945a0SNathan Whitehorn init_dialog(FILE *input, FILE *output)
3294c8945a0SNathan Whitehorn {
3304c8945a0SNathan Whitehorn int fd1, fd2;
3314c8945a0SNathan Whitehorn char *device = 0;
3324c8945a0SNathan Whitehorn
3332a3e3873SBaptiste Daroussin setlocale(LC_ALL, "");
3342a3e3873SBaptiste Daroussin
3354c8945a0SNathan Whitehorn dialog_state.output = output;
336a96ef450SBaptiste Daroussin if (dialog_state.tab_len == 0)
3374c8945a0SNathan Whitehorn dialog_state.tab_len = TAB_LEN;
338a96ef450SBaptiste Daroussin if (dialog_state.aspect_ratio == 0)
3394c8945a0SNathan Whitehorn dialog_state.aspect_ratio = DEFAULT_ASPECT_RATIO;
3404c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
3414c8945a0SNathan Whitehorn dialog_state.use_colors = USE_COLORS; /* use colors by default? */
3424c8945a0SNathan Whitehorn dialog_state.use_shadow = USE_SHADOW; /* shadow dialog boxes by default? */
3434c8945a0SNathan Whitehorn #endif
3444c8945a0SNathan Whitehorn
3454c8945a0SNathan Whitehorn #ifdef HAVE_RC_FILE
3464c8945a0SNathan Whitehorn if (dlg_parse_rc() == -1) /* Read the configuration file */
3474c8945a0SNathan Whitehorn dlg_exiterr("init_dialog: dlg_parse_rc");
3484c8945a0SNathan Whitehorn #endif
3494c8945a0SNathan Whitehorn
3504c8945a0SNathan Whitehorn /*
3514c8945a0SNathan Whitehorn * Some widgets (such as gauge) may read from the standard input. Pipes
3524c8945a0SNathan Whitehorn * only connect stdout/stdin, so there is not much choice. But reading a
3534c8945a0SNathan Whitehorn * pipe would get in the way of curses' normal reading stdin for getch.
3544c8945a0SNathan Whitehorn *
3554c8945a0SNathan Whitehorn * As in the --stdout (see below), reopening the terminal does not always
3564c8945a0SNathan Whitehorn * work properly. dialog provides a --pipe-fd option for this purpose. We
3574c8945a0SNathan Whitehorn * test that case first (differing fileno's for input/stdin). If the
3584c8945a0SNathan Whitehorn * fileno's are equal, but we're not reading from a tty, see if we can open
3594c8945a0SNathan Whitehorn * /dev/tty.
3604c8945a0SNathan Whitehorn */
3614c8945a0SNathan Whitehorn dialog_state.pipe_input = stdin;
3624c8945a0SNathan Whitehorn if (fileno(input) != fileno(stdin)) {
36319718649SNathan Whitehorn if ((fd1 = dup(fileno(input))) >= 0
3644c8945a0SNathan Whitehorn && (fd2 = dup(fileno(stdin))) >= 0) {
3654c8945a0SNathan Whitehorn (void) dup2(fileno(input), fileno(stdin));
3664c8945a0SNathan Whitehorn dialog_state.pipe_input = fdopen(fd2, "r");
3674c8945a0SNathan Whitehorn if (fileno(stdin) != 0) /* some functions may read fd #0 */
3684c8945a0SNathan Whitehorn (void) dup2(fileno(stdin), 0);
3692a3e3873SBaptiste Daroussin } else {
3704c8945a0SNathan Whitehorn dlg_exiterr("cannot open tty-input");
3712a3e3873SBaptiste Daroussin }
3722a3e3873SBaptiste Daroussin close(fd1);
3734c8945a0SNathan Whitehorn } else if (!isatty(fileno(stdin))) {
3742a3e3873SBaptiste Daroussin if ((fd1 = open_terminal(&device, O_RDONLY)) >= 0) {
3752a3e3873SBaptiste Daroussin if ((fd2 = dup(fileno(stdin))) >= 0) {
3764c8945a0SNathan Whitehorn dialog_state.pipe_input = fdopen(fd2, "r");
3774c8945a0SNathan Whitehorn if (freopen(device, "r", stdin) == 0)
3784c8945a0SNathan Whitehorn dlg_exiterr("cannot open tty-input");
3794c8945a0SNathan Whitehorn if (fileno(stdin) != 0) /* some functions may read fd #0 */
3804c8945a0SNathan Whitehorn (void) dup2(fileno(stdin), 0);
3814c8945a0SNathan Whitehorn }
3822a3e3873SBaptiste Daroussin close(fd1);
3832a3e3873SBaptiste Daroussin }
3844c8945a0SNathan Whitehorn free(device);
3854c8945a0SNathan Whitehorn }
3864c8945a0SNathan Whitehorn
3874c8945a0SNathan Whitehorn /*
3884c8945a0SNathan Whitehorn * If stdout is not a tty and dialog is called with the --stdout option, we
3894c8945a0SNathan Whitehorn * have to provide for a way to write to the screen.
3904c8945a0SNathan Whitehorn *
3914c8945a0SNathan Whitehorn * The curses library normally writes its output to stdout, leaving stderr
3924c8945a0SNathan Whitehorn * free for scripting. Scripts are simpler when stdout is redirected. The
3934c8945a0SNathan Whitehorn * newterm function is useful; it allows us to specify where the output
3944c8945a0SNathan Whitehorn * goes. Reopening the terminal is not portable since several
3954c8945a0SNathan Whitehorn * configurations do not allow this to work properly:
3964c8945a0SNathan Whitehorn *
3974c8945a0SNathan Whitehorn * a) some getty implementations (and possibly broken tty drivers, e.g., on
3984c8945a0SNathan Whitehorn * HPUX 10 and 11) cause stdin to act as if it is still in cooked mode
3994c8945a0SNathan Whitehorn * even though results from ioctl's state that it is successfully
4004c8945a0SNathan Whitehorn * altered to raw mode. Broken is the proper term.
4014c8945a0SNathan Whitehorn *
4024c8945a0SNathan Whitehorn * b) the user may not have permissions on the device, e.g., if one su's
4034c8945a0SNathan Whitehorn * from the login user to another non-privileged user.
4044c8945a0SNathan Whitehorn */
4054c8945a0SNathan Whitehorn if (!isatty(fileno(stdout))
4064c8945a0SNathan Whitehorn && (fileno(stdout) == fileno(output) || dialog_tty())) {
4074c8945a0SNathan Whitehorn if ((fd1 = open_terminal(&device, O_WRONLY)) >= 0
4084c8945a0SNathan Whitehorn && (dialog_state.screen_output = fdopen(fd1, "w")) != 0) {
4094c8945a0SNathan Whitehorn if (newterm(NULL, dialog_state.screen_output, stdin) == 0) {
4104c8945a0SNathan Whitehorn dlg_exiterr("cannot initialize curses");
4114c8945a0SNathan Whitehorn }
4124c8945a0SNathan Whitehorn free(device);
4134c8945a0SNathan Whitehorn } else {
4144c8945a0SNathan Whitehorn dlg_exiterr("cannot open tty-output");
4154c8945a0SNathan Whitehorn }
4164c8945a0SNathan Whitehorn } else {
4174c8945a0SNathan Whitehorn dialog_state.screen_output = stdout;
4184c8945a0SNathan Whitehorn (void) initscr();
4194c8945a0SNathan Whitehorn }
420a96ef450SBaptiste Daroussin dlg_keep_tite(dialog_state.screen_output);
4214c8945a0SNathan Whitehorn #ifdef HAVE_FLUSHINP
4224c8945a0SNathan Whitehorn (void) flushinp();
4234c8945a0SNathan Whitehorn #endif
4244c8945a0SNathan Whitehorn (void) keypad(stdscr, TRUE);
4254c8945a0SNathan Whitehorn (void) cbreak();
4264c8945a0SNathan Whitehorn (void) noecho();
4277a1c0d96SNathan Whitehorn
4287a1c0d96SNathan Whitehorn if (!dialog_state.no_mouse) {
4294c8945a0SNathan Whitehorn mouse_open();
4307a1c0d96SNathan Whitehorn }
4317a1c0d96SNathan Whitehorn
4324c8945a0SNathan Whitehorn dialog_state.screen_initialized = TRUE;
4334c8945a0SNathan Whitehorn
4344c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
4354c8945a0SNathan Whitehorn if (dialog_state.use_colors || dialog_state.use_shadow)
4364c8945a0SNathan Whitehorn dlg_color_setup(); /* Set up colors */
4374c8945a0SNathan Whitehorn #endif
4384c8945a0SNathan Whitehorn
4394c8945a0SNathan Whitehorn /* Set screen to screen attribute */
4404c8945a0SNathan Whitehorn dlg_clear();
4414c8945a0SNathan Whitehorn }
4424c8945a0SNathan Whitehorn
443a96ef450SBaptiste Daroussin void
dlg_keep_tite(FILE * output)444a96ef450SBaptiste Daroussin dlg_keep_tite(FILE *output)
445a96ef450SBaptiste Daroussin {
446a96ef450SBaptiste Daroussin if (!dialog_vars.keep_tite) {
447a96ef450SBaptiste Daroussin #if CAN_KEEP_TITE
448a96ef450SBaptiste Daroussin /*
449a96ef450SBaptiste Daroussin * Cancel xterm's alternate-screen mode.
450a96ef450SBaptiste Daroussin */
451a96ef450SBaptiste Daroussin if ((fileno(output) != fileno(stdout)
452a96ef450SBaptiste Daroussin || isatty(fileno(output)))
453a96ef450SBaptiste Daroussin && key_mouse != 0 /* xterm and kindred */
454a96ef450SBaptiste Daroussin && isprivate(enter_ca_mode)
455a96ef450SBaptiste Daroussin && isprivate(exit_ca_mode)) {
456a96ef450SBaptiste Daroussin FILE *save = dialog_state.screen_output;
457a96ef450SBaptiste Daroussin
458a96ef450SBaptiste Daroussin /*
459a96ef450SBaptiste Daroussin * initscr() or newterm() already wrote enter_ca_mode as a side
460a96ef450SBaptiste Daroussin * effect of initializing the screen. It would be nice to not even
461a96ef450SBaptiste Daroussin * do that, but we do not really have access to the correct copy of
462a96ef450SBaptiste Daroussin * the terminfo description until those functions have been
463a96ef450SBaptiste Daroussin * invoked.
464a96ef450SBaptiste Daroussin */
465a96ef450SBaptiste Daroussin (void) refresh();
466a96ef450SBaptiste Daroussin dialog_state.screen_output = output;
467a96ef450SBaptiste Daroussin (void) tputs(exit_ca_mode, 0, my_putc);
468a96ef450SBaptiste Daroussin (void) tputs(clear_screen, 0, my_putc);
469a96ef450SBaptiste Daroussin dialog_state.screen_output = save;
470a96ef450SBaptiste Daroussin
471a96ef450SBaptiste Daroussin /*
472a96ef450SBaptiste Daroussin * Prevent ncurses from switching "back" to the normal screen when
473a96ef450SBaptiste Daroussin * exiting from dialog. That would move the cursor to the original
474a96ef450SBaptiste Daroussin * location saved in xterm. Normally curses sets the cursor
475a96ef450SBaptiste Daroussin * position to the first line after the display, but the alternate
476a96ef450SBaptiste Daroussin * screen switching is done after that point.
477a96ef450SBaptiste Daroussin *
478a96ef450SBaptiste Daroussin * Cancelling the strings altogether also works around the buggy
479a96ef450SBaptiste Daroussin * implementation of alternate-screen in rxvt, etc., which clear
480a96ef450SBaptiste Daroussin * more of the display than they should.
481a96ef450SBaptiste Daroussin */
482a96ef450SBaptiste Daroussin enter_ca_mode = 0;
483a96ef450SBaptiste Daroussin exit_ca_mode = 0;
484a96ef450SBaptiste Daroussin }
485a96ef450SBaptiste Daroussin #else
486a96ef450SBaptiste Daroussin /*
487a96ef450SBaptiste Daroussin * For other implementations, there are no useful answers:
488a96ef450SBaptiste Daroussin * + SVr4 curses "could" support a similar approach, but the clue about
489a96ef450SBaptiste Daroussin * xterm is absent from its terminal database.
490a96ef450SBaptiste Daroussin * + PDCurses does not provide terminfo.
491a96ef450SBaptiste Daroussin */
492a96ef450SBaptiste Daroussin (void) output;
493a96ef450SBaptiste Daroussin #endif
494a96ef450SBaptiste Daroussin }
495a96ef450SBaptiste Daroussin }
496a96ef450SBaptiste Daroussin
4974c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
4984c8945a0SNathan Whitehorn static int defined_colors = 1; /* pair-0 is reserved */
4994c8945a0SNathan Whitehorn /*
5004c8945a0SNathan Whitehorn * Setup for color display
5014c8945a0SNathan Whitehorn */
5024c8945a0SNathan Whitehorn void
dlg_color_setup(void)5034c8945a0SNathan Whitehorn dlg_color_setup(void)
5044c8945a0SNathan Whitehorn {
505a96ef450SBaptiste Daroussin if (has_colors()) { /* Terminal supports color? */
5064c8945a0SNathan Whitehorn unsigned i;
5074c8945a0SNathan Whitehorn
5084c8945a0SNathan Whitehorn (void) start_color();
5094c8945a0SNathan Whitehorn
5104c8945a0SNathan Whitehorn #if defined(HAVE_USE_DEFAULT_COLORS)
5114c8945a0SNathan Whitehorn use_default_colors();
5124c8945a0SNathan Whitehorn #endif
5134c8945a0SNathan Whitehorn
5144c8945a0SNathan Whitehorn #if defined(__NetBSD__) && defined(_CURSES_)
5154c8945a0SNathan Whitehorn #define C_ATTR(x,y) (((x) != 0 ? A_BOLD : 0) | COLOR_PAIR((y)))
5164c8945a0SNathan Whitehorn /* work around bug in NetBSD curses */
5174c8945a0SNathan Whitehorn for (i = 0; i < sizeof(dlg_color_table) /
5184c8945a0SNathan Whitehorn sizeof(dlg_color_table[0]); i++) {
5194c8945a0SNathan Whitehorn
5204c8945a0SNathan Whitehorn /* Initialize color pairs */
5214c8945a0SNathan Whitehorn (void) init_pair(i + 1,
5224c8945a0SNathan Whitehorn dlg_color_table[i].fg,
5234c8945a0SNathan Whitehorn dlg_color_table[i].bg);
5244c8945a0SNathan Whitehorn
5254c8945a0SNathan Whitehorn /* Setup color attributes */
5264c8945a0SNathan Whitehorn dlg_color_table[i].atr = C_ATTR(dlg_color_table[i].hilite, i + 1);
5274c8945a0SNathan Whitehorn }
5284c8945a0SNathan Whitehorn defined_colors = i + 1;
5294c8945a0SNathan Whitehorn #else
5304c8945a0SNathan Whitehorn for (i = 0; i < sizeof(dlg_color_table) /
5314c8945a0SNathan Whitehorn sizeof(dlg_color_table[0]); i++) {
5324c8945a0SNathan Whitehorn
5334c8945a0SNathan Whitehorn /* Initialize color pairs */
534a96ef450SBaptiste Daroussin chtype atr = dlg_color_pair(dlg_color_table[i].fg,
5354c8945a0SNathan Whitehorn dlg_color_table[i].bg);
5364c8945a0SNathan Whitehorn
537a96ef450SBaptiste Daroussin atr |= (dlg_color_table[i].hilite ? A_BOLD : 0);
538a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
539a96ef450SBaptiste Daroussin atr |= (dlg_color_table[i].ul ? A_UNDERLINE : 0);
540a96ef450SBaptiste Daroussin atr |= (dlg_color_table[i].rv ? A_REVERSE : 0);
541a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
542a96ef450SBaptiste Daroussin
543a96ef450SBaptiste Daroussin dlg_color_table[i].atr = atr;
5444c8945a0SNathan Whitehorn }
5454c8945a0SNathan Whitehorn #endif
5464c8945a0SNathan Whitehorn } else {
5474c8945a0SNathan Whitehorn dialog_state.use_colors = FALSE;
5484c8945a0SNathan Whitehorn dialog_state.use_shadow = FALSE;
5494c8945a0SNathan Whitehorn }
5504c8945a0SNathan Whitehorn }
5514c8945a0SNathan Whitehorn
5524c8945a0SNathan Whitehorn int
dlg_color_count(void)5534c8945a0SNathan Whitehorn dlg_color_count(void)
5544c8945a0SNathan Whitehorn {
555a96ef450SBaptiste Daroussin return TableSize(dlg_color_table);
5564c8945a0SNathan Whitehorn }
5574c8945a0SNathan Whitehorn
5584c8945a0SNathan Whitehorn /*
5597a1c0d96SNathan Whitehorn * Wrapper for getattrs(), or the more cumbersome X/Open wattr_get().
5607a1c0d96SNathan Whitehorn */
5617a1c0d96SNathan Whitehorn chtype
dlg_get_attrs(WINDOW * win)5627a1c0d96SNathan Whitehorn dlg_get_attrs(WINDOW *win)
5637a1c0d96SNathan Whitehorn {
5647a1c0d96SNathan Whitehorn chtype result;
5657a1c0d96SNathan Whitehorn #ifdef HAVE_GETATTRS
566682c9e0fSNathan Whitehorn result = (chtype) getattrs(win);
5677a1c0d96SNathan Whitehorn #else
5687a1c0d96SNathan Whitehorn attr_t my_result;
5697a1c0d96SNathan Whitehorn short my_pair;
5707a1c0d96SNathan Whitehorn wattr_get(win, &my_result, &my_pair, NULL);
5717a1c0d96SNathan Whitehorn result = my_result;
5727a1c0d96SNathan Whitehorn #endif
5737a1c0d96SNathan Whitehorn return result;
5747a1c0d96SNathan Whitehorn }
5757a1c0d96SNathan Whitehorn
5767a1c0d96SNathan Whitehorn /*
5774c8945a0SNathan Whitehorn * Reuse color pairs (they are limited), returning a COLOR_PAIR() value if we
5784c8945a0SNathan Whitehorn * have (or can) define a pair with the given color as foreground on the
5794c8945a0SNathan Whitehorn * window's defined background.
5804c8945a0SNathan Whitehorn */
5814c8945a0SNathan Whitehorn chtype
dlg_color_pair(int foreground,int background)5824c8945a0SNathan Whitehorn dlg_color_pair(int foreground, int background)
5834c8945a0SNathan Whitehorn {
5844c8945a0SNathan Whitehorn chtype result = 0;
5854c8945a0SNathan Whitehorn int pair;
5864c8945a0SNathan Whitehorn short fg, bg;
5874c8945a0SNathan Whitehorn bool found = FALSE;
5884c8945a0SNathan Whitehorn
5894c8945a0SNathan Whitehorn for (pair = 1; pair < defined_colors; ++pair) {
5904c8945a0SNathan Whitehorn if (pair_content((short) pair, &fg, &bg) != ERR
5914c8945a0SNathan Whitehorn && fg == foreground
5924c8945a0SNathan Whitehorn && bg == background) {
5934c8945a0SNathan Whitehorn result = (chtype) COLOR_PAIR(pair);
5944c8945a0SNathan Whitehorn found = TRUE;
5954c8945a0SNathan Whitehorn break;
5964c8945a0SNathan Whitehorn }
5974c8945a0SNathan Whitehorn }
5984c8945a0SNathan Whitehorn if (!found && (defined_colors + 1) < COLOR_PAIRS) {
5994c8945a0SNathan Whitehorn pair = defined_colors++;
6004c8945a0SNathan Whitehorn (void) init_pair((short) pair, (short) foreground, (short) background);
6014c8945a0SNathan Whitehorn result = (chtype) COLOR_PAIR(pair);
6024c8945a0SNathan Whitehorn }
6034c8945a0SNathan Whitehorn return result;
6044c8945a0SNathan Whitehorn }
6054c8945a0SNathan Whitehorn
6064c8945a0SNathan Whitehorn /*
6074c8945a0SNathan Whitehorn * Reuse color pairs (they are limited), returning a COLOR_PAIR() value if we
6084c8945a0SNathan Whitehorn * have (or can) define a pair with the given color as foreground on the
6094c8945a0SNathan Whitehorn * window's defined background.
6104c8945a0SNathan Whitehorn */
6114c8945a0SNathan Whitehorn static chtype
define_color(WINDOW * win,int foreground)6124c8945a0SNathan Whitehorn define_color(WINDOW *win, int foreground)
6134c8945a0SNathan Whitehorn {
6144c8945a0SNathan Whitehorn short fg, bg, background;
615f4f33ea0SBaptiste Daroussin if (dialog_state.text_only) {
616f4f33ea0SBaptiste Daroussin background = COLOR_BLACK;
617f4f33ea0SBaptiste Daroussin } else {
618f4f33ea0SBaptiste Daroussin chtype attrs = dlg_get_attrs(win);
619a96ef450SBaptiste Daroussin int pair;
6204c8945a0SNathan Whitehorn
6214c8945a0SNathan Whitehorn if ((pair = PAIR_NUMBER(attrs)) != 0
6224c8945a0SNathan Whitehorn && pair_content((short) pair, &fg, &bg) != ERR) {
6234c8945a0SNathan Whitehorn background = bg;
6244c8945a0SNathan Whitehorn } else {
6254c8945a0SNathan Whitehorn background = COLOR_BLACK;
6264c8945a0SNathan Whitehorn }
627f4f33ea0SBaptiste Daroussin }
6284c8945a0SNathan Whitehorn return dlg_color_pair(foreground, background);
6294c8945a0SNathan Whitehorn }
6304c8945a0SNathan Whitehorn #endif
6314c8945a0SNathan Whitehorn
6324c8945a0SNathan Whitehorn /*
6334c8945a0SNathan Whitehorn * End using dialog functions.
6344c8945a0SNathan Whitehorn */
6354c8945a0SNathan Whitehorn void
end_dialog(void)6364c8945a0SNathan Whitehorn end_dialog(void)
6374c8945a0SNathan Whitehorn {
6384c8945a0SNathan Whitehorn if (dialog_state.screen_initialized) {
6394c8945a0SNathan Whitehorn dialog_state.screen_initialized = FALSE;
640a96ef450SBaptiste Daroussin if (dialog_vars.erase_on_exit) {
641a96ef450SBaptiste Daroussin /*
642a96ef450SBaptiste Daroussin * Clear the screen to the native background color, and leave the
643a96ef450SBaptiste Daroussin * terminal cursor at the lower-left corner of the screen.
644a96ef450SBaptiste Daroussin */
645a96ef450SBaptiste Daroussin werase(stdscr);
646a96ef450SBaptiste Daroussin wrefresh(stdscr);
647a96ef450SBaptiste Daroussin }
6484c8945a0SNathan Whitehorn mouse_close();
6494c8945a0SNathan Whitehorn (void) endwin();
6504c8945a0SNathan Whitehorn (void) fflush(stdout);
6514c8945a0SNathan Whitehorn }
6524c8945a0SNathan Whitehorn }
6534c8945a0SNathan Whitehorn
654682c9e0fSNathan Whitehorn #define ESCAPE_LEN 3
6554c8945a0SNathan Whitehorn #define isOurEscape(p) (((p)[0] == '\\') && ((p)[1] == 'Z') && ((p)[2] != 0))
6564c8945a0SNathan Whitehorn
6572a3e3873SBaptiste Daroussin int
dlg_count_real_columns(const char * text)6582a3e3873SBaptiste Daroussin dlg_count_real_columns(const char *text)
6592a3e3873SBaptiste Daroussin {
660febdb468SDevin Teske int result = 0;
661febdb468SDevin Teske if (*text) {
662febdb468SDevin Teske result = dlg_count_columns(text);
6632a3e3873SBaptiste Daroussin if (result && dialog_vars.colors) {
6642a3e3873SBaptiste Daroussin int hidden = 0;
6652a3e3873SBaptiste Daroussin while (*text) {
6662a3e3873SBaptiste Daroussin if (dialog_vars.colors && isOurEscape(text)) {
6672a3e3873SBaptiste Daroussin hidden += ESCAPE_LEN;
6682a3e3873SBaptiste Daroussin text += ESCAPE_LEN;
6692a3e3873SBaptiste Daroussin } else {
6702a3e3873SBaptiste Daroussin ++text;
6712a3e3873SBaptiste Daroussin }
6722a3e3873SBaptiste Daroussin }
6732a3e3873SBaptiste Daroussin result -= hidden;
6742a3e3873SBaptiste Daroussin }
675febdb468SDevin Teske }
6762a3e3873SBaptiste Daroussin return result;
6772a3e3873SBaptiste Daroussin }
6782a3e3873SBaptiste Daroussin
6794c8945a0SNathan Whitehorn static int
centered(int width,const char * string)6804c8945a0SNathan Whitehorn centered(int width, const char *string)
6814c8945a0SNathan Whitehorn {
6822a3e3873SBaptiste Daroussin int need = dlg_count_real_columns(string);
6834c8945a0SNathan Whitehorn int left;
6844c8945a0SNathan Whitehorn
6852a3e3873SBaptiste Daroussin left = (width - need) / 2 - 1;
6864c8945a0SNathan Whitehorn if (left < 0)
6874c8945a0SNathan Whitehorn left = 0;
6884c8945a0SNathan Whitehorn return left;
6894c8945a0SNathan Whitehorn }
6904c8945a0SNathan Whitehorn
6917a1c0d96SNathan Whitehorn #ifdef USE_WIDE_CURSES
6927a1c0d96SNathan Whitehorn static bool
is_combining(const char * txt,int * combined)6937a1c0d96SNathan Whitehorn is_combining(const char *txt, int *combined)
6947a1c0d96SNathan Whitehorn {
6957a1c0d96SNathan Whitehorn bool result = FALSE;
6967a1c0d96SNathan Whitehorn
6977a1c0d96SNathan Whitehorn if (*combined == 0) {
6987a1c0d96SNathan Whitehorn if (UCH(*txt) >= 128) {
6997a1c0d96SNathan Whitehorn wchar_t wch;
7007a1c0d96SNathan Whitehorn mbstate_t state;
7017a1c0d96SNathan Whitehorn size_t given = strlen(txt);
7027a1c0d96SNathan Whitehorn size_t len;
7037a1c0d96SNathan Whitehorn
7047a1c0d96SNathan Whitehorn memset(&state, 0, sizeof(state));
7057a1c0d96SNathan Whitehorn len = mbrtowc(&wch, txt, given, &state);
7067a1c0d96SNathan Whitehorn if ((int) len > 0 && wcwidth(wch) == 0) {
7077a1c0d96SNathan Whitehorn *combined = (int) len - 1;
7087a1c0d96SNathan Whitehorn result = TRUE;
7097a1c0d96SNathan Whitehorn }
7107a1c0d96SNathan Whitehorn }
7117a1c0d96SNathan Whitehorn } else {
7127a1c0d96SNathan Whitehorn result = TRUE;
7137a1c0d96SNathan Whitehorn *combined -= 1;
7147a1c0d96SNathan Whitehorn }
7157a1c0d96SNathan Whitehorn return result;
7167a1c0d96SNathan Whitehorn }
7177a1c0d96SNathan Whitehorn #endif
7187a1c0d96SNathan Whitehorn
7194c8945a0SNathan Whitehorn /*
7202a3e3873SBaptiste Daroussin * Print the name (tag) or text from a DIALOG_LISTITEM, highlighting the
7212a3e3873SBaptiste Daroussin * first character if selected.
7222a3e3873SBaptiste Daroussin */
7232a3e3873SBaptiste Daroussin void
dlg_print_listitem(WINDOW * win,const char * text,int climit,bool first,int selected)7242a3e3873SBaptiste Daroussin dlg_print_listitem(WINDOW *win,
7252a3e3873SBaptiste Daroussin const char *text,
7262a3e3873SBaptiste Daroussin int climit,
7272a3e3873SBaptiste Daroussin bool first,
7282a3e3873SBaptiste Daroussin int selected)
7292a3e3873SBaptiste Daroussin {
7302a3e3873SBaptiste Daroussin chtype attr = A_NORMAL;
7312a3e3873SBaptiste Daroussin int limit;
7322a3e3873SBaptiste Daroussin chtype attrs[4];
7332a3e3873SBaptiste Daroussin
7342a3e3873SBaptiste Daroussin if (text == 0)
7352a3e3873SBaptiste Daroussin text = "";
7362a3e3873SBaptiste Daroussin
737a96ef450SBaptiste Daroussin if (first && !dialog_vars.no_hot_list) {
7382a3e3873SBaptiste Daroussin const int *indx = dlg_index_wchars(text);
7392a3e3873SBaptiste Daroussin attrs[3] = tag_key_selected_attr;
7402a3e3873SBaptiste Daroussin attrs[2] = tag_key_attr;
7412a3e3873SBaptiste Daroussin attrs[1] = tag_selected_attr;
7422a3e3873SBaptiste Daroussin attrs[0] = tag_attr;
7432a3e3873SBaptiste Daroussin
744f4f33ea0SBaptiste Daroussin dlg_attrset(win, selected ? attrs[3] : attrs[2]);
745a96ef450SBaptiste Daroussin if (*text != '\0') {
7462a3e3873SBaptiste Daroussin (void) waddnstr(win, text, indx[1]);
7472a3e3873SBaptiste Daroussin
7482a3e3873SBaptiste Daroussin if ((int) strlen(text) > indx[1]) {
7492a3e3873SBaptiste Daroussin limit = dlg_limit_columns(text, climit, 1);
7502a3e3873SBaptiste Daroussin if (limit > 1) {
751f4f33ea0SBaptiste Daroussin dlg_attrset(win, selected ? attrs[1] : attrs[0]);
7522a3e3873SBaptiste Daroussin (void) waddnstr(win,
7532a3e3873SBaptiste Daroussin text + indx[1],
7542a3e3873SBaptiste Daroussin indx[limit] - indx[1]);
7552a3e3873SBaptiste Daroussin }
7562a3e3873SBaptiste Daroussin }
757a96ef450SBaptiste Daroussin }
7582a3e3873SBaptiste Daroussin } else {
759a96ef450SBaptiste Daroussin const int *cols;
760a96ef450SBaptiste Daroussin
7612a3e3873SBaptiste Daroussin attrs[1] = item_selected_attr;
7622a3e3873SBaptiste Daroussin attrs[0] = item_attr;
7632a3e3873SBaptiste Daroussin
7642a3e3873SBaptiste Daroussin cols = dlg_index_columns(text);
7652a3e3873SBaptiste Daroussin limit = dlg_limit_columns(text, climit, 0);
7662a3e3873SBaptiste Daroussin
7672a3e3873SBaptiste Daroussin if (limit > 0) {
768f4f33ea0SBaptiste Daroussin dlg_attrset(win, selected ? attrs[1] : attrs[0]);
7692a3e3873SBaptiste Daroussin dlg_print_text(win, text, cols[limit], &attr);
7702a3e3873SBaptiste Daroussin }
7712a3e3873SBaptiste Daroussin }
7722a3e3873SBaptiste Daroussin }
7732a3e3873SBaptiste Daroussin
7742a3e3873SBaptiste Daroussin /*
7754c8945a0SNathan Whitehorn * Print up to 'cols' columns from 'text', optionally rendering our escape
7764c8945a0SNathan Whitehorn * sequence for attributes and color.
7774c8945a0SNathan Whitehorn */
7784c8945a0SNathan Whitehorn void
dlg_print_text(WINDOW * win,const char * txt,int cols,chtype * attr)7794c8945a0SNathan Whitehorn dlg_print_text(WINDOW *win, const char *txt, int cols, chtype *attr)
7804c8945a0SNathan Whitehorn {
7814c8945a0SNathan Whitehorn int y_origin, x_origin;
7824c8945a0SNathan Whitehorn int y_before, x_before = 0;
7834c8945a0SNathan Whitehorn int y_after, x_after;
7844c8945a0SNathan Whitehorn int tabbed = 0;
7854c8945a0SNathan Whitehorn bool ended = FALSE;
7867a1c0d96SNathan Whitehorn #ifdef USE_WIDE_CURSES
7877a1c0d96SNathan Whitehorn int combined = 0;
7887a1c0d96SNathan Whitehorn #endif
7894c8945a0SNathan Whitehorn
790f4f33ea0SBaptiste Daroussin if (dialog_state.text_only) {
791f4f33ea0SBaptiste Daroussin y_origin = y_after = 0;
792f4f33ea0SBaptiste Daroussin x_origin = x_after = 0;
793f4f33ea0SBaptiste Daroussin } else {
794a96ef450SBaptiste Daroussin y_after = 0;
795a96ef450SBaptiste Daroussin x_after = 0;
7964c8945a0SNathan Whitehorn getyx(win, y_origin, x_origin);
797f4f33ea0SBaptiste Daroussin }
7984c8945a0SNathan Whitehorn while (cols > 0 && (*txt != '\0')) {
799a96ef450SBaptiste Daroussin bool thisTab;
800a96ef450SBaptiste Daroussin chtype useattr;
801a96ef450SBaptiste Daroussin
8024c8945a0SNathan Whitehorn if (dialog_vars.colors) {
8034c8945a0SNathan Whitehorn while (isOurEscape(txt)) {
8044c8945a0SNathan Whitehorn int code;
8054c8945a0SNathan Whitehorn
8064c8945a0SNathan Whitehorn txt += 2;
8074c8945a0SNathan Whitehorn switch (code = CharOf(*txt)) {
8084c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
8094c8945a0SNathan Whitehorn case '0':
8104c8945a0SNathan Whitehorn case '1':
8114c8945a0SNathan Whitehorn case '2':
8124c8945a0SNathan Whitehorn case '3':
8134c8945a0SNathan Whitehorn case '4':
8144c8945a0SNathan Whitehorn case '5':
8154c8945a0SNathan Whitehorn case '6':
8164c8945a0SNathan Whitehorn case '7':
8174c8945a0SNathan Whitehorn *attr &= ~A_COLOR;
8184c8945a0SNathan Whitehorn *attr |= define_color(win, code - '0');
8194c8945a0SNathan Whitehorn break;
8204c8945a0SNathan Whitehorn #endif
8214c8945a0SNathan Whitehorn case 'B':
8224c8945a0SNathan Whitehorn *attr &= ~A_BOLD;
8234c8945a0SNathan Whitehorn break;
8244c8945a0SNathan Whitehorn case 'b':
8254c8945a0SNathan Whitehorn *attr |= A_BOLD;
8264c8945a0SNathan Whitehorn break;
8274c8945a0SNathan Whitehorn case 'R':
8284c8945a0SNathan Whitehorn *attr &= ~A_REVERSE;
8294c8945a0SNathan Whitehorn break;
8304c8945a0SNathan Whitehorn case 'r':
8314c8945a0SNathan Whitehorn *attr |= A_REVERSE;
8324c8945a0SNathan Whitehorn break;
8334c8945a0SNathan Whitehorn case 'U':
8344c8945a0SNathan Whitehorn *attr &= ~A_UNDERLINE;
8354c8945a0SNathan Whitehorn break;
8364c8945a0SNathan Whitehorn case 'u':
8374c8945a0SNathan Whitehorn *attr |= A_UNDERLINE;
8384c8945a0SNathan Whitehorn break;
8394c8945a0SNathan Whitehorn case 'n':
8404c8945a0SNathan Whitehorn *attr = A_NORMAL;
8414c8945a0SNathan Whitehorn break;
842a96ef450SBaptiste Daroussin default:
843a96ef450SBaptiste Daroussin break;
8444c8945a0SNathan Whitehorn }
8454c8945a0SNathan Whitehorn ++txt;
8464c8945a0SNathan Whitehorn }
8474c8945a0SNathan Whitehorn }
8484c8945a0SNathan Whitehorn if (ended || *txt == '\n' || *txt == '\0')
8494c8945a0SNathan Whitehorn break;
8504c8945a0SNathan Whitehorn useattr = (*attr) & A_ATTRIBUTES;
8514c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
8524c8945a0SNathan Whitehorn /*
8534c8945a0SNathan Whitehorn * Prevent this from making text invisible when the foreground and
8544c8945a0SNathan Whitehorn * background colors happen to be the same, and there's no bold
8554c8945a0SNathan Whitehorn * attribute.
8564c8945a0SNathan Whitehorn */
8574c8945a0SNathan Whitehorn if ((useattr & A_COLOR) != 0 && (useattr & A_BOLD) == 0) {
8584c8945a0SNathan Whitehorn short pair = (short) PAIR_NUMBER(useattr);
8594c8945a0SNathan Whitehorn short fg, bg;
8604c8945a0SNathan Whitehorn if (pair_content(pair, &fg, &bg) != ERR
8614c8945a0SNathan Whitehorn && fg == bg) {
8624c8945a0SNathan Whitehorn useattr &= ~A_COLOR;
8634c8945a0SNathan Whitehorn useattr |= dlg_color_pair(fg, ((bg == COLOR_BLACK)
8644c8945a0SNathan Whitehorn ? COLOR_WHITE
8654c8945a0SNathan Whitehorn : COLOR_BLACK));
8664c8945a0SNathan Whitehorn }
8674c8945a0SNathan Whitehorn }
8684c8945a0SNathan Whitehorn #endif
8694c8945a0SNathan Whitehorn /*
8704c8945a0SNathan Whitehorn * Write the character, using curses to tell exactly how wide it
8714c8945a0SNathan Whitehorn * is. If it is a tab, discount that, since the caller thinks
8724c8945a0SNathan Whitehorn * tabs are nonprinting, and curses will expand tabs to one or
8734c8945a0SNathan Whitehorn * more blanks.
8744c8945a0SNathan Whitehorn */
8754c8945a0SNathan Whitehorn thisTab = (CharOf(*txt) == TAB);
876f4f33ea0SBaptiste Daroussin if (dialog_state.text_only) {
877f4f33ea0SBaptiste Daroussin x_before = x_after;
878f4f33ea0SBaptiste Daroussin } else {
8792a3e3873SBaptiste Daroussin if (thisTab) {
8804c8945a0SNathan Whitehorn getyx(win, y_before, x_before);
8812a3e3873SBaptiste Daroussin (void) y_before;
8822a3e3873SBaptiste Daroussin }
883f4f33ea0SBaptiste Daroussin }
884f4f33ea0SBaptiste Daroussin if (dialog_state.text_only) {
885f4f33ea0SBaptiste Daroussin int ch = CharOf(*txt++);
886f4f33ea0SBaptiste Daroussin if (thisTab) {
887f4f33ea0SBaptiste Daroussin while ((x_after++) % 8) {
888f4f33ea0SBaptiste Daroussin fputc(' ', dialog_state.output);
889f4f33ea0SBaptiste Daroussin }
890f4f33ea0SBaptiste Daroussin } else {
891f4f33ea0SBaptiste Daroussin fputc(ch, dialog_state.output);
892f4f33ea0SBaptiste Daroussin x_after++; /* FIXME: handle meta per locale */
893f4f33ea0SBaptiste Daroussin }
894f4f33ea0SBaptiste Daroussin } else {
8954c8945a0SNathan Whitehorn (void) waddch(win, CharOf(*txt++) | useattr);
8964c8945a0SNathan Whitehorn getyx(win, y_after, x_after);
897f4f33ea0SBaptiste Daroussin }
8984c8945a0SNathan Whitehorn if (thisTab && (y_after == y_origin))
8994c8945a0SNathan Whitehorn tabbed += (x_after - x_before);
9007a1c0d96SNathan Whitehorn if ((y_after != y_origin) ||
9017a1c0d96SNathan Whitehorn (x_after >= (cols + tabbed + x_origin)
9027a1c0d96SNathan Whitehorn #ifdef USE_WIDE_CURSES
9037a1c0d96SNathan Whitehorn && !is_combining(txt, &combined)
9047a1c0d96SNathan Whitehorn #endif
9057a1c0d96SNathan Whitehorn )) {
9064c8945a0SNathan Whitehorn ended = TRUE;
9074c8945a0SNathan Whitehorn }
9084c8945a0SNathan Whitehorn }
909f4f33ea0SBaptiste Daroussin if (dialog_state.text_only) {
910f4f33ea0SBaptiste Daroussin fputc('\n', dialog_state.output);
911f4f33ea0SBaptiste Daroussin }
9124c8945a0SNathan Whitehorn }
9134c8945a0SNathan Whitehorn
9144c8945a0SNathan Whitehorn /*
9154c8945a0SNathan Whitehorn * Print one line of the prompt in the window within the limits of the
9164c8945a0SNathan Whitehorn * specified right margin. The line will end on a word boundary and a pointer
9174c8945a0SNathan Whitehorn * to the start of the next line is returned, or a NULL pointer if the end of
9184c8945a0SNathan Whitehorn * *prompt is reached.
9194c8945a0SNathan Whitehorn */
9204c8945a0SNathan Whitehorn const char *
dlg_print_line(WINDOW * win,chtype * attr,const char * prompt,int lm,int rm,int * x)9214c8945a0SNathan Whitehorn dlg_print_line(WINDOW *win,
9224c8945a0SNathan Whitehorn chtype *attr,
9234c8945a0SNathan Whitehorn const char *prompt,
9244c8945a0SNathan Whitehorn int lm, int rm, int *x)
9254c8945a0SNathan Whitehorn {
9262a3e3873SBaptiste Daroussin const char *wrap_ptr;
9272a3e3873SBaptiste Daroussin const char *test_ptr;
928682c9e0fSNathan Whitehorn const char *hide_ptr = 0;
9294c8945a0SNathan Whitehorn const int *cols = dlg_index_columns(prompt);
9304c8945a0SNathan Whitehorn const int *indx = dlg_index_wchars(prompt);
9314c8945a0SNathan Whitehorn int wrap_inx = 0;
9324c8945a0SNathan Whitehorn int test_inx = 0;
9334c8945a0SNathan Whitehorn int cur_x = lm;
9344c8945a0SNathan Whitehorn int hidden = 0;
9354c8945a0SNathan Whitehorn int limit = dlg_count_wchars(prompt);
9364c8945a0SNathan Whitehorn int n;
9374c8945a0SNathan Whitehorn int tabbed = 0;
9384c8945a0SNathan Whitehorn
9394c8945a0SNathan Whitehorn *x = 1;
9404c8945a0SNathan Whitehorn
9414c8945a0SNathan Whitehorn /*
9424c8945a0SNathan Whitehorn * Set *test_ptr to the end of the line or the right margin (rm), whichever
9434c8945a0SNathan Whitehorn * is less, and set wrap_ptr to the end of the last word in the line.
9444c8945a0SNathan Whitehorn */
9454c8945a0SNathan Whitehorn for (n = 0; n < limit; ++n) {
946f4f33ea0SBaptiste Daroussin int ch = *(test_ptr = prompt + indx[test_inx]);
947f4f33ea0SBaptiste Daroussin if (ch == '\n' || ch == '\0' || cur_x >= (rm + hidden))
9484c8945a0SNathan Whitehorn break;
949f4f33ea0SBaptiste Daroussin if (ch == TAB && n == 0) {
9504c8945a0SNathan Whitehorn tabbed = 8; /* workaround for leading tabs */
951f4f33ea0SBaptiste Daroussin } else if (isblank(UCH(ch))
952f4f33ea0SBaptiste Daroussin && n != 0
953f4f33ea0SBaptiste Daroussin && !isblank(UCH(prompt[indx[n - 1]]))) {
9544c8945a0SNathan Whitehorn wrap_inx = n;
9554c8945a0SNathan Whitehorn *x = cur_x;
9562a3e3873SBaptiste Daroussin } else if (dialog_vars.colors && isOurEscape(test_ptr)) {
957682c9e0fSNathan Whitehorn hide_ptr = test_ptr;
958682c9e0fSNathan Whitehorn hidden += ESCAPE_LEN;
959682c9e0fSNathan Whitehorn n += (ESCAPE_LEN - 1);
9604c8945a0SNathan Whitehorn }
9614c8945a0SNathan Whitehorn cur_x = lm + tabbed + cols[n + 1];
9624c8945a0SNathan Whitehorn if (cur_x > (rm + hidden))
9634c8945a0SNathan Whitehorn break;
9644c8945a0SNathan Whitehorn test_inx = n + 1;
9654c8945a0SNathan Whitehorn }
9664c8945a0SNathan Whitehorn
9674c8945a0SNathan Whitehorn /*
9684c8945a0SNathan Whitehorn * If the line doesn't reach the right margin in the middle of a word, then
9694c8945a0SNathan Whitehorn * we don't have to wrap it at the end of the previous word.
9704c8945a0SNathan Whitehorn */
9714c8945a0SNathan Whitehorn test_ptr = prompt + indx[test_inx];
972f4f33ea0SBaptiste Daroussin if (*test_ptr == '\n' || isblank(UCH(*test_ptr)) || *test_ptr == '\0') {
9734c8945a0SNathan Whitehorn wrap_inx = test_inx;
974f4f33ea0SBaptiste Daroussin while (wrap_inx > 0 && isblank(UCH(prompt[indx[wrap_inx - 1]]))) {
9754c8945a0SNathan Whitehorn wrap_inx--;
9764c8945a0SNathan Whitehorn }
9774c8945a0SNathan Whitehorn *x = lm + indx[wrap_inx];
9784c8945a0SNathan Whitehorn } else if (*x == 1 && cur_x >= rm) {
9794c8945a0SNathan Whitehorn /*
9804c8945a0SNathan Whitehorn * If the line has no spaces, then wrap it anyway at the right margin
9814c8945a0SNathan Whitehorn */
9824c8945a0SNathan Whitehorn *x = rm;
9834c8945a0SNathan Whitehorn wrap_inx = test_inx;
9844c8945a0SNathan Whitehorn }
9854c8945a0SNathan Whitehorn wrap_ptr = prompt + indx[wrap_inx];
9867a1c0d96SNathan Whitehorn #ifdef USE_WIDE_CURSES
9877a1c0d96SNathan Whitehorn if (UCH(*wrap_ptr) >= 128) {
9887a1c0d96SNathan Whitehorn int combined = 0;
9897a1c0d96SNathan Whitehorn while (is_combining(wrap_ptr, &combined)) {
9907a1c0d96SNathan Whitehorn ++wrap_ptr;
9917a1c0d96SNathan Whitehorn }
9927a1c0d96SNathan Whitehorn }
9937a1c0d96SNathan Whitehorn #endif
9944c8945a0SNathan Whitehorn
9954c8945a0SNathan Whitehorn /*
996682c9e0fSNathan Whitehorn * If we found hidden text past the last point that we will display,
997682c9e0fSNathan Whitehorn * discount that from the displayed length.
998682c9e0fSNathan Whitehorn */
999682c9e0fSNathan Whitehorn if ((hide_ptr != 0) && (hide_ptr >= wrap_ptr)) {
1000682c9e0fSNathan Whitehorn hidden -= ESCAPE_LEN;
1001682c9e0fSNathan Whitehorn test_ptr = wrap_ptr;
1002682c9e0fSNathan Whitehorn while (test_ptr < wrap_ptr) {
10032a3e3873SBaptiste Daroussin if (dialog_vars.colors && isOurEscape(test_ptr)) {
1004682c9e0fSNathan Whitehorn hidden -= ESCAPE_LEN;
1005682c9e0fSNathan Whitehorn test_ptr += ESCAPE_LEN;
1006682c9e0fSNathan Whitehorn } else {
1007682c9e0fSNathan Whitehorn ++test_ptr;
1008682c9e0fSNathan Whitehorn }
1009682c9e0fSNathan Whitehorn }
1010682c9e0fSNathan Whitehorn }
1011682c9e0fSNathan Whitehorn
1012682c9e0fSNathan Whitehorn /*
10134c8945a0SNathan Whitehorn * Print the line if we have a window pointer. Otherwise this routine
10144c8945a0SNathan Whitehorn * is just being called for sizing the window.
10154c8945a0SNathan Whitehorn */
1016f4f33ea0SBaptiste Daroussin if (dialog_state.text_only || win) {
10174c8945a0SNathan Whitehorn dlg_print_text(win, prompt, (cols[wrap_inx] - hidden), attr);
10184c8945a0SNathan Whitehorn }
10194c8945a0SNathan Whitehorn
10204c8945a0SNathan Whitehorn /* *x tells the calling function how long the line was */
1021f4f33ea0SBaptiste Daroussin if (*x == 1) {
10224c8945a0SNathan Whitehorn *x = rm;
1023f4f33ea0SBaptiste Daroussin }
10244c8945a0SNathan Whitehorn
1025682c9e0fSNathan Whitehorn *x -= hidden;
1026682c9e0fSNathan Whitehorn
10274c8945a0SNathan Whitehorn /* Find the start of the next line and return a pointer to it */
10284c8945a0SNathan Whitehorn test_ptr = wrap_ptr;
1029f4f33ea0SBaptiste Daroussin while (isblank(UCH(*test_ptr)))
10304c8945a0SNathan Whitehorn test_ptr++;
10314c8945a0SNathan Whitehorn if (*test_ptr == '\n')
10324c8945a0SNathan Whitehorn test_ptr++;
1033febdb468SDevin Teske dlg_finish_string(prompt);
10344c8945a0SNathan Whitehorn return (test_ptr);
10354c8945a0SNathan Whitehorn }
10364c8945a0SNathan Whitehorn
10374c8945a0SNathan Whitehorn static void
justify_text(WINDOW * win,const char * prompt,int limit_y,int limit_x,int * high,int * wide)10384c8945a0SNathan Whitehorn justify_text(WINDOW *win,
10394c8945a0SNathan Whitehorn const char *prompt,
10404c8945a0SNathan Whitehorn int limit_y,
10414c8945a0SNathan Whitehorn int limit_x,
10424c8945a0SNathan Whitehorn int *high, int *wide)
10434c8945a0SNathan Whitehorn {
10444c8945a0SNathan Whitehorn chtype attr = A_NORMAL;
1045a96ef450SBaptiste Daroussin int x;
10464c8945a0SNathan Whitehorn int y = MARGIN;
10474c8945a0SNathan Whitehorn int max_x = 2;
10484c8945a0SNathan Whitehorn int lm = (2 * MARGIN); /* left margin (box-border plus a space) */
10494c8945a0SNathan Whitehorn int rm = limit_x; /* right margin */
10504c8945a0SNathan Whitehorn int bm = limit_y; /* bottom margin */
10514c8945a0SNathan Whitehorn int last_y = 0, last_x = 0;
10524c8945a0SNathan Whitehorn
1053f4f33ea0SBaptiste Daroussin dialog_state.text_height = 0;
1054f4f33ea0SBaptiste Daroussin dialog_state.text_width = 0;
1055f4f33ea0SBaptiste Daroussin if (dialog_state.text_only || win) {
10564c8945a0SNathan Whitehorn rm -= (2 * MARGIN);
10574c8945a0SNathan Whitehorn bm -= (2 * MARGIN);
10584c8945a0SNathan Whitehorn }
10594c8945a0SNathan Whitehorn if (prompt == 0)
10604c8945a0SNathan Whitehorn prompt = "";
10614c8945a0SNathan Whitehorn
10624c8945a0SNathan Whitehorn if (win != 0)
10634c8945a0SNathan Whitehorn getyx(win, last_y, last_x);
10644c8945a0SNathan Whitehorn while (y <= bm && *prompt) {
10654c8945a0SNathan Whitehorn x = lm;
10664c8945a0SNathan Whitehorn
10674c8945a0SNathan Whitehorn if (*prompt == '\n') {
10684c8945a0SNathan Whitehorn while (*prompt == '\n' && y < bm) {
10694c8945a0SNathan Whitehorn if (*(prompt + 1) != '\0') {
10704c8945a0SNathan Whitehorn ++y;
10714c8945a0SNathan Whitehorn if (win != 0)
10724c8945a0SNathan Whitehorn (void) wmove(win, y, lm);
10734c8945a0SNathan Whitehorn }
10744c8945a0SNathan Whitehorn prompt++;
10754c8945a0SNathan Whitehorn }
10764c8945a0SNathan Whitehorn } else if (win != 0)
10774c8945a0SNathan Whitehorn (void) wmove(win, y, lm);
10784c8945a0SNathan Whitehorn
10794c8945a0SNathan Whitehorn if (*prompt) {
10804c8945a0SNathan Whitehorn prompt = dlg_print_line(win, &attr, prompt, lm, rm, &x);
10814c8945a0SNathan Whitehorn if (win != 0)
10824c8945a0SNathan Whitehorn getyx(win, last_y, last_x);
10834c8945a0SNathan Whitehorn }
10844c8945a0SNathan Whitehorn if (*prompt) {
10854c8945a0SNathan Whitehorn ++y;
10864c8945a0SNathan Whitehorn if (win != 0)
10874c8945a0SNathan Whitehorn (void) wmove(win, y, lm);
10884c8945a0SNathan Whitehorn }
10894c8945a0SNathan Whitehorn max_x = MAX(max_x, x);
10904c8945a0SNathan Whitehorn }
10914c8945a0SNathan Whitehorn /* Move back to the last position after drawing prompt, for msgbox. */
10924c8945a0SNathan Whitehorn if (win != 0)
10934c8945a0SNathan Whitehorn (void) wmove(win, last_y, last_x);
10944c8945a0SNathan Whitehorn
10954c8945a0SNathan Whitehorn /* Set the final height and width for the calling function */
10964c8945a0SNathan Whitehorn if (high != 0)
10974c8945a0SNathan Whitehorn *high = y;
10984c8945a0SNathan Whitehorn if (wide != 0)
10994c8945a0SNathan Whitehorn *wide = max_x;
11004c8945a0SNathan Whitehorn }
11014c8945a0SNathan Whitehorn
11024c8945a0SNathan Whitehorn /*
11034c8945a0SNathan Whitehorn * Print a string of text in a window, automatically wrap around to the next
11044c8945a0SNathan Whitehorn * line if the string is too long to fit on one line. Note that the string may
11054c8945a0SNathan Whitehorn * contain embedded newlines.
11064c8945a0SNathan Whitehorn */
11074c8945a0SNathan Whitehorn void
dlg_print_autowrap(WINDOW * win,const char * prompt,int height,int width)11084c8945a0SNathan Whitehorn dlg_print_autowrap(WINDOW *win, const char *prompt, int height, int width)
11094c8945a0SNathan Whitehorn {
11104c8945a0SNathan Whitehorn justify_text(win, prompt,
11114c8945a0SNathan Whitehorn height,
11124c8945a0SNathan Whitehorn width,
11134c8945a0SNathan Whitehorn (int *) 0, (int *) 0);
11144c8945a0SNathan Whitehorn }
11154c8945a0SNathan Whitehorn
11164c8945a0SNathan Whitehorn /*
11174c8945a0SNathan Whitehorn * Display the message in a scrollable window. Actually the way it works is
11184c8945a0SNathan Whitehorn * that we create a "tall" window of the proper width, let the text wrap within
11194c8945a0SNathan Whitehorn * that, and copy a slice of the result to the dialog.
11204c8945a0SNathan Whitehorn *
11214c8945a0SNathan Whitehorn * It works for ncurses. Other curses implementations show only blanks (Tru64)
11224c8945a0SNathan Whitehorn * or garbage (NetBSD).
11234c8945a0SNathan Whitehorn */
11244c8945a0SNathan Whitehorn int
dlg_print_scrolled(WINDOW * win,const char * prompt,int offset,int height,int width,int pauseopt)11254c8945a0SNathan Whitehorn dlg_print_scrolled(WINDOW *win,
11264c8945a0SNathan Whitehorn const char *prompt,
11274c8945a0SNathan Whitehorn int offset,
11284c8945a0SNathan Whitehorn int height,
11294c8945a0SNathan Whitehorn int width,
11304c8945a0SNathan Whitehorn int pauseopt)
11314c8945a0SNathan Whitehorn {
11324c8945a0SNathan Whitehorn int oldy, oldx;
11334c8945a0SNathan Whitehorn int last = 0;
11344c8945a0SNathan Whitehorn
11357a1c0d96SNathan Whitehorn (void) pauseopt; /* used only for ncurses */
11367a1c0d96SNathan Whitehorn
11374c8945a0SNathan Whitehorn getyx(win, oldy, oldx);
11384c8945a0SNathan Whitehorn #ifdef NCURSES_VERSION
11394c8945a0SNathan Whitehorn if (pauseopt) {
11404c8945a0SNathan Whitehorn int wide = width - (2 * MARGIN);
11414c8945a0SNathan Whitehorn int high = LINES;
11424c8945a0SNathan Whitehorn int len;
11434c8945a0SNathan Whitehorn WINDOW *dummy;
11444c8945a0SNathan Whitehorn
11454c8945a0SNathan Whitehorn #if defined(NCURSES_VERSION_PATCH) && NCURSES_VERSION_PATCH >= 20040417
11464c8945a0SNathan Whitehorn /*
11474c8945a0SNathan Whitehorn * If we're not limited by the screensize, allow text to possibly be
11484c8945a0SNathan Whitehorn * one character per line.
11494c8945a0SNathan Whitehorn */
11504c8945a0SNathan Whitehorn if ((len = dlg_count_columns(prompt)) > high)
11514c8945a0SNathan Whitehorn high = len;
11524c8945a0SNathan Whitehorn #endif
11534c8945a0SNathan Whitehorn dummy = newwin(high, width, 0, 0);
1154682c9e0fSNathan Whitehorn if (dummy == 0) {
1155f4f33ea0SBaptiste Daroussin dlg_attrset(win, dialog_attr);
1156682c9e0fSNathan Whitehorn dlg_print_autowrap(win, prompt, height + 1 + (3 * MARGIN), width);
1157682c9e0fSNathan Whitehorn last = 0;
1158682c9e0fSNathan Whitehorn } else {
1159a96ef450SBaptiste Daroussin int y, x;
1160a96ef450SBaptiste Daroussin
11614c8945a0SNathan Whitehorn wbkgdset(dummy, dialog_attr | ' ');
1162f4f33ea0SBaptiste Daroussin dlg_attrset(dummy, dialog_attr);
11634c8945a0SNathan Whitehorn werase(dummy);
11644c8945a0SNathan Whitehorn dlg_print_autowrap(dummy, prompt, high, width);
11654c8945a0SNathan Whitehorn getyx(dummy, y, x);
11662a3e3873SBaptiste Daroussin (void) x;
11674c8945a0SNathan Whitehorn
11684c8945a0SNathan Whitehorn copywin(dummy, /* srcwin */
11694c8945a0SNathan Whitehorn win, /* dstwin */
11704c8945a0SNathan Whitehorn offset + MARGIN, /* sminrow */
11714c8945a0SNathan Whitehorn MARGIN, /* smincol */
11724c8945a0SNathan Whitehorn MARGIN, /* dminrow */
11734c8945a0SNathan Whitehorn MARGIN, /* dmincol */
11744c8945a0SNathan Whitehorn height, /* dmaxrow */
11754c8945a0SNathan Whitehorn wide, /* dmaxcol */
11764c8945a0SNathan Whitehorn FALSE);
11774c8945a0SNathan Whitehorn
11784c8945a0SNathan Whitehorn delwin(dummy);
11794c8945a0SNathan Whitehorn
11804c8945a0SNathan Whitehorn /* if the text is incomplete, or we have scrolled, show the percentage */
11814c8945a0SNathan Whitehorn if (y > 0 && wide > 4) {
1182a96ef450SBaptiste Daroussin int percent = (int) ((height + offset) * 100.0 / y);
1183a96ef450SBaptiste Daroussin
11844c8945a0SNathan Whitehorn if (percent < 0)
11854c8945a0SNathan Whitehorn percent = 0;
11864c8945a0SNathan Whitehorn if (percent > 100)
11874c8945a0SNathan Whitehorn percent = 100;
1188a96ef450SBaptiste Daroussin
11894c8945a0SNathan Whitehorn if (offset != 0 || percent != 100) {
1190a96ef450SBaptiste Daroussin char buffer[5];
1191a96ef450SBaptiste Daroussin
1192f4f33ea0SBaptiste Daroussin dlg_attrset(win, position_indicator_attr);
11934c8945a0SNathan Whitehorn (void) wmove(win, MARGIN + height, wide - 4);
11944c8945a0SNathan Whitehorn (void) sprintf(buffer, "%d%%", percent);
11954c8945a0SNathan Whitehorn (void) waddstr(win, buffer);
11964c8945a0SNathan Whitehorn if ((len = (int) strlen(buffer)) < 4) {
1197f4f33ea0SBaptiste Daroussin dlg_attrset(win, border_attr);
11984c8945a0SNathan Whitehorn whline(win, dlg_boxchar(ACS_HLINE), 4 - len);
11994c8945a0SNathan Whitehorn }
12004c8945a0SNathan Whitehorn }
12014c8945a0SNathan Whitehorn }
12024c8945a0SNathan Whitehorn last = (y - height);
1203682c9e0fSNathan Whitehorn }
12044c8945a0SNathan Whitehorn } else
12054c8945a0SNathan Whitehorn #endif
12064c8945a0SNathan Whitehorn {
12074c8945a0SNathan Whitehorn (void) offset;
1208f4f33ea0SBaptiste Daroussin dlg_attrset(win, dialog_attr);
12094c8945a0SNathan Whitehorn dlg_print_autowrap(win, prompt, height + 1 + (3 * MARGIN), width);
12104c8945a0SNathan Whitehorn last = 0;
12114c8945a0SNathan Whitehorn }
12124c8945a0SNathan Whitehorn wmove(win, oldy, oldx);
12134c8945a0SNathan Whitehorn return last;
12144c8945a0SNathan Whitehorn }
12154c8945a0SNathan Whitehorn
12164c8945a0SNathan Whitehorn int
dlg_check_scrolled(int key,int last,int page,bool * show,int * offset)12174c8945a0SNathan Whitehorn dlg_check_scrolled(int key, int last, int page, bool * show, int *offset)
12184c8945a0SNathan Whitehorn {
12194c8945a0SNathan Whitehorn int code = 0;
12204c8945a0SNathan Whitehorn
12214c8945a0SNathan Whitehorn *show = FALSE;
12224c8945a0SNathan Whitehorn
12234c8945a0SNathan Whitehorn switch (key) {
12244c8945a0SNathan Whitehorn case DLGK_PAGE_FIRST:
12254c8945a0SNathan Whitehorn if (*offset > 0) {
12264c8945a0SNathan Whitehorn *offset = 0;
12274c8945a0SNathan Whitehorn *show = TRUE;
12284c8945a0SNathan Whitehorn }
12294c8945a0SNathan Whitehorn break;
12304c8945a0SNathan Whitehorn case DLGK_PAGE_LAST:
12314c8945a0SNathan Whitehorn if (*offset < last) {
12324c8945a0SNathan Whitehorn *offset = last;
12334c8945a0SNathan Whitehorn *show = TRUE;
12344c8945a0SNathan Whitehorn }
12354c8945a0SNathan Whitehorn break;
12364c8945a0SNathan Whitehorn case DLGK_GRID_UP:
12374c8945a0SNathan Whitehorn if (*offset > 0) {
12384c8945a0SNathan Whitehorn --(*offset);
12394c8945a0SNathan Whitehorn *show = TRUE;
12404c8945a0SNathan Whitehorn }
12414c8945a0SNathan Whitehorn break;
12424c8945a0SNathan Whitehorn case DLGK_GRID_DOWN:
12434c8945a0SNathan Whitehorn if (*offset < last) {
12444c8945a0SNathan Whitehorn ++(*offset);
12454c8945a0SNathan Whitehorn *show = TRUE;
12464c8945a0SNathan Whitehorn }
12474c8945a0SNathan Whitehorn break;
12484c8945a0SNathan Whitehorn case DLGK_PAGE_PREV:
12494c8945a0SNathan Whitehorn if (*offset > 0) {
12504c8945a0SNathan Whitehorn *offset -= page;
12514c8945a0SNathan Whitehorn if (*offset < 0)
12524c8945a0SNathan Whitehorn *offset = 0;
12534c8945a0SNathan Whitehorn *show = TRUE;
12544c8945a0SNathan Whitehorn }
12554c8945a0SNathan Whitehorn break;
12564c8945a0SNathan Whitehorn case DLGK_PAGE_NEXT:
12574c8945a0SNathan Whitehorn if (*offset < last) {
12584c8945a0SNathan Whitehorn *offset += page;
12594c8945a0SNathan Whitehorn if (*offset > last)
12604c8945a0SNathan Whitehorn *offset = last;
12614c8945a0SNathan Whitehorn *show = TRUE;
12624c8945a0SNathan Whitehorn }
12634c8945a0SNathan Whitehorn break;
12644c8945a0SNathan Whitehorn default:
12654c8945a0SNathan Whitehorn code = -1;
12664c8945a0SNathan Whitehorn break;
12674c8945a0SNathan Whitehorn }
12684c8945a0SNathan Whitehorn return code;
12694c8945a0SNathan Whitehorn }
12704c8945a0SNathan Whitehorn
12714c8945a0SNathan Whitehorn /*
12724c8945a0SNathan Whitehorn * Calculate the window size for preformatted text. This will calculate box
12734c8945a0SNathan Whitehorn * dimensions that are at or close to the specified aspect ratio for the prompt
12744c8945a0SNathan Whitehorn * string with all spaces and newlines preserved and additional newlines added
12754c8945a0SNathan Whitehorn * as necessary.
12764c8945a0SNathan Whitehorn */
12774c8945a0SNathan Whitehorn static void
auto_size_preformatted(const char * prompt,int * height,int * width)12784c8945a0SNathan Whitehorn auto_size_preformatted(const char *prompt, int *height, int *width)
12794c8945a0SNathan Whitehorn {
12804c8945a0SNathan Whitehorn int high = 0, wide = 0;
12814c8945a0SNathan Whitehorn float car; /* Calculated Aspect Ratio */
12824c8945a0SNathan Whitehorn int max_y = SLINES - 1;
12834c8945a0SNathan Whitehorn int max_x = SCOLS - 2;
12844c8945a0SNathan Whitehorn int max_width = max_x;
12854c8945a0SNathan Whitehorn int ar = dialog_state.aspect_ratio;
12864c8945a0SNathan Whitehorn
12874c8945a0SNathan Whitehorn /* Get the initial dimensions */
12884c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
12894c8945a0SNathan Whitehorn car = (float) (wide / high);
12904c8945a0SNathan Whitehorn
12914c8945a0SNathan Whitehorn /*
12924c8945a0SNathan Whitehorn * If the aspect ratio is greater than it should be, then decrease the
12934c8945a0SNathan Whitehorn * width proportionately.
12944c8945a0SNathan Whitehorn */
12954c8945a0SNathan Whitehorn if (car > ar) {
1296a96ef450SBaptiste Daroussin float diff = car / (float) ar;
12974c8945a0SNathan Whitehorn max_x = (int) ((float) wide / diff + 4);
12984c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
12994c8945a0SNathan Whitehorn car = (float) wide / (float) high;
13004c8945a0SNathan Whitehorn }
13014c8945a0SNathan Whitehorn
13024c8945a0SNathan Whitehorn /*
13034c8945a0SNathan Whitehorn * If the aspect ratio is too small after decreasing the width, then
13044c8945a0SNathan Whitehorn * incrementally increase the width until the aspect ratio is equal to or
13054c8945a0SNathan Whitehorn * greater than the specified aspect ratio.
13064c8945a0SNathan Whitehorn */
13074c8945a0SNathan Whitehorn while (car < ar && max_x < max_width) {
13084c8945a0SNathan Whitehorn max_x += 4;
13094c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
13104c8945a0SNathan Whitehorn car = (float) (wide / high);
13114c8945a0SNathan Whitehorn }
13124c8945a0SNathan Whitehorn
13134c8945a0SNathan Whitehorn *height = high;
13144c8945a0SNathan Whitehorn *width = wide;
13154c8945a0SNathan Whitehorn }
13164c8945a0SNathan Whitehorn
13174c8945a0SNathan Whitehorn /*
13184c8945a0SNathan Whitehorn * Find the length of the longest "word" in the given string. By setting the
13194c8945a0SNathan Whitehorn * widget width at least this long, we can avoid splitting a word on the
13204c8945a0SNathan Whitehorn * margin.
13214c8945a0SNathan Whitehorn */
13224c8945a0SNathan Whitehorn static int
longest_word(const char * string)13234c8945a0SNathan Whitehorn longest_word(const char *string)
13244c8945a0SNathan Whitehorn {
1325a96ef450SBaptiste Daroussin int result = 0;
13264c8945a0SNathan Whitehorn
13274c8945a0SNathan Whitehorn while (*string != '\0') {
1328a96ef450SBaptiste Daroussin int length = 0;
13294c8945a0SNathan Whitehorn while (*string != '\0' && !isspace(UCH(*string))) {
13304c8945a0SNathan Whitehorn length++;
13314c8945a0SNathan Whitehorn string++;
13324c8945a0SNathan Whitehorn }
13334c8945a0SNathan Whitehorn result = MAX(result, length);
13344c8945a0SNathan Whitehorn if (*string != '\0')
13354c8945a0SNathan Whitehorn string++;
13364c8945a0SNathan Whitehorn }
13374c8945a0SNathan Whitehorn return result;
13384c8945a0SNathan Whitehorn }
13394c8945a0SNathan Whitehorn
13404c8945a0SNathan Whitehorn /*
13414c8945a0SNathan Whitehorn * if (height or width == -1) Maximize()
13424c8945a0SNathan Whitehorn * if (height or width == 0), justify and return actual limits.
13434c8945a0SNathan Whitehorn */
13444c8945a0SNathan Whitehorn static void
real_auto_size(const char * title,const char * prompt,int * height,int * width,int boxlines,int mincols)13454c8945a0SNathan Whitehorn real_auto_size(const char *title,
13464c8945a0SNathan Whitehorn const char *prompt,
13474c8945a0SNathan Whitehorn int *height, int *width,
13484c8945a0SNathan Whitehorn int boxlines, int mincols)
13494c8945a0SNathan Whitehorn {
13504c8945a0SNathan Whitehorn int x = (dialog_vars.begin_set ? dialog_vars.begin_x : 2);
13514c8945a0SNathan Whitehorn int y = (dialog_vars.begin_set ? dialog_vars.begin_y : 1);
13524c8945a0SNathan Whitehorn int title_length = title ? dlg_count_columns(title) : 0;
13534c8945a0SNathan Whitehorn int high;
13544c8945a0SNathan Whitehorn int save_high = *height;
13554c8945a0SNathan Whitehorn int save_wide = *width;
1356f4f33ea0SBaptiste Daroussin int max_high;
1357f4f33ea0SBaptiste Daroussin int max_wide;
13584c8945a0SNathan Whitehorn
13594c8945a0SNathan Whitehorn if (prompt == 0) {
13604c8945a0SNathan Whitehorn if (*height == 0)
13614c8945a0SNathan Whitehorn *height = -1;
13624c8945a0SNathan Whitehorn if (*width == 0)
13634c8945a0SNathan Whitehorn *width = -1;
13644c8945a0SNathan Whitehorn }
13654c8945a0SNathan Whitehorn
1366f4f33ea0SBaptiste Daroussin max_high = (*height < 0);
1367f4f33ea0SBaptiste Daroussin max_wide = (*width < 0);
1368f4f33ea0SBaptiste Daroussin
13694c8945a0SNathan Whitehorn if (*height > 0) {
13704c8945a0SNathan Whitehorn high = *height;
13714c8945a0SNathan Whitehorn } else {
13724c8945a0SNathan Whitehorn high = SLINES - y;
13734c8945a0SNathan Whitehorn }
13744c8945a0SNathan Whitehorn
13752a3e3873SBaptiste Daroussin if (*width <= 0) {
1376a96ef450SBaptiste Daroussin int wide;
1377a96ef450SBaptiste Daroussin
13782a3e3873SBaptiste Daroussin if (prompt != 0) {
13794c8945a0SNathan Whitehorn wide = MAX(title_length, mincols);
13804c8945a0SNathan Whitehorn if (strchr(prompt, '\n') == 0) {
13812a3e3873SBaptiste Daroussin double val = (dialog_state.aspect_ratio *
13822a3e3873SBaptiste Daroussin dlg_count_real_columns(prompt));
13834c8945a0SNathan Whitehorn double xxx = sqrt(val);
13844c8945a0SNathan Whitehorn int tmp = (int) xxx;
13854c8945a0SNathan Whitehorn wide = MAX(wide, tmp);
13864c8945a0SNathan Whitehorn wide = MAX(wide, longest_word(prompt));
13874c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, high, wide, height, width);
13884c8945a0SNathan Whitehorn } else {
13894c8945a0SNathan Whitehorn auto_size_preformatted(prompt, height, width);
13904c8945a0SNathan Whitehorn }
13914c8945a0SNathan Whitehorn } else {
13924c8945a0SNathan Whitehorn wide = SCOLS - x;
13934c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, high, wide, height, width);
13944c8945a0SNathan Whitehorn }
13952a3e3873SBaptiste Daroussin }
13964c8945a0SNathan Whitehorn
13974c8945a0SNathan Whitehorn if (*width < title_length) {
13984c8945a0SNathan Whitehorn justify_text((WINDOW *) 0, prompt, high, title_length, height, width);
13994c8945a0SNathan Whitehorn *width = title_length;
14004c8945a0SNathan Whitehorn }
14014c8945a0SNathan Whitehorn
1402f4f33ea0SBaptiste Daroussin dialog_state.text_height = *height;
1403f4f33ea0SBaptiste Daroussin dialog_state.text_width = *width;
1404f4f33ea0SBaptiste Daroussin
14054c8945a0SNathan Whitehorn if (*width < mincols && save_wide == 0)
14064c8945a0SNathan Whitehorn *width = mincols;
14074c8945a0SNathan Whitehorn if (prompt != 0) {
1408f4f33ea0SBaptiste Daroussin *width += ((2 * MARGIN) + SHADOW_COLS);
1409f4f33ea0SBaptiste Daroussin *height += boxlines + (2 * MARGIN);
14104c8945a0SNathan Whitehorn }
1411f4f33ea0SBaptiste Daroussin
14124c8945a0SNathan Whitehorn if (save_high > 0)
14134c8945a0SNathan Whitehorn *height = save_high;
14144c8945a0SNathan Whitehorn if (save_wide > 0)
14154c8945a0SNathan Whitehorn *width = save_wide;
1416f4f33ea0SBaptiste Daroussin
1417f4f33ea0SBaptiste Daroussin if (max_high)
1418f4f33ea0SBaptiste Daroussin *height = SLINES - (dialog_vars.begin_set ? dialog_vars.begin_y : 0);
1419f4f33ea0SBaptiste Daroussin if (max_wide)
1420f4f33ea0SBaptiste Daroussin *width = SCOLS - (dialog_vars.begin_set ? dialog_vars.begin_x : 0);
14214c8945a0SNathan Whitehorn }
14224c8945a0SNathan Whitehorn
14234c8945a0SNathan Whitehorn /* End of real_auto_size() */
14244c8945a0SNathan Whitehorn
14254c8945a0SNathan Whitehorn void
dlg_auto_size(const char * title,const char * prompt,int * height,int * width,int boxlines,int mincols)14264c8945a0SNathan Whitehorn dlg_auto_size(const char *title,
14274c8945a0SNathan Whitehorn const char *prompt,
14284c8945a0SNathan Whitehorn int *height,
14294c8945a0SNathan Whitehorn int *width,
14304c8945a0SNathan Whitehorn int boxlines,
14314c8945a0SNathan Whitehorn int mincols)
14324c8945a0SNathan Whitehorn {
1433f4f33ea0SBaptiste Daroussin DLG_TRACE(("# dlg_auto_size(%d,%d) limits %d,%d\n",
1434f4f33ea0SBaptiste Daroussin *height, *width,
1435f4f33ea0SBaptiste Daroussin boxlines, mincols));
1436f4f33ea0SBaptiste Daroussin
14374c8945a0SNathan Whitehorn real_auto_size(title, prompt, height, width, boxlines, mincols);
14384c8945a0SNathan Whitehorn
14394c8945a0SNathan Whitehorn if (*width > SCOLS) {
14404c8945a0SNathan Whitehorn (*height)++;
14414c8945a0SNathan Whitehorn *width = SCOLS;
14424c8945a0SNathan Whitehorn }
14434c8945a0SNathan Whitehorn
1444f4f33ea0SBaptiste Daroussin if (*height > SLINES) {
14454c8945a0SNathan Whitehorn *height = SLINES;
14464c8945a0SNathan Whitehorn }
1447f4f33ea0SBaptiste Daroussin DLG_TRACE(("# ...dlg_auto_size(%d,%d) also %d,%d\n",
1448f4f33ea0SBaptiste Daroussin *height, *width,
1449f4f33ea0SBaptiste Daroussin dialog_state.text_height, dialog_state.text_width));
1450f4f33ea0SBaptiste Daroussin }
14514c8945a0SNathan Whitehorn
14524c8945a0SNathan Whitehorn /*
14534c8945a0SNathan Whitehorn * if (height or width == -1) Maximize()
14544c8945a0SNathan Whitehorn * if (height or width == 0)
14554c8945a0SNathan Whitehorn * height=MIN(SLINES, num.lines in fd+n);
14564c8945a0SNathan Whitehorn * width=MIN(SCOLS, MAX(longer line+n, mincols));
14574c8945a0SNathan Whitehorn */
14584c8945a0SNathan Whitehorn void
dlg_auto_sizefile(const char * title,const char * file,int * height,int * width,int boxlines,int mincols)14594c8945a0SNathan Whitehorn dlg_auto_sizefile(const char *title,
14604c8945a0SNathan Whitehorn const char *file,
14614c8945a0SNathan Whitehorn int *height,
14624c8945a0SNathan Whitehorn int *width,
14634c8945a0SNathan Whitehorn int boxlines,
14644c8945a0SNathan Whitehorn int mincols)
14654c8945a0SNathan Whitehorn {
14664c8945a0SNathan Whitehorn int count = 0;
14674c8945a0SNathan Whitehorn int len = title ? dlg_count_columns(title) : 0;
14684c8945a0SNathan Whitehorn int nc = 4;
14694c8945a0SNathan Whitehorn int numlines = 2;
14704c8945a0SNathan Whitehorn FILE *fd;
14714c8945a0SNathan Whitehorn
14724c8945a0SNathan Whitehorn /* Open input file for reading */
14734c8945a0SNathan Whitehorn if ((fd = fopen(file, "rb")) == NULL)
14744c8945a0SNathan Whitehorn dlg_exiterr("dlg_auto_sizefile: Cannot open input file %s", file);
14754c8945a0SNathan Whitehorn
14764c8945a0SNathan Whitehorn if ((*height == -1) || (*width == -1)) {
14774c8945a0SNathan Whitehorn *height = SLINES - (dialog_vars.begin_set ? dialog_vars.begin_y : 0);
14784c8945a0SNathan Whitehorn *width = SCOLS - (dialog_vars.begin_set ? dialog_vars.begin_x : 0);
14794c8945a0SNathan Whitehorn }
14804c8945a0SNathan Whitehorn if ((*height != 0) && (*width != 0)) {
14814c8945a0SNathan Whitehorn (void) fclose(fd);
14824c8945a0SNathan Whitehorn if (*width > SCOLS)
14834c8945a0SNathan Whitehorn *width = SCOLS;
14844c8945a0SNathan Whitehorn if (*height > SLINES)
14854c8945a0SNathan Whitehorn *height = SLINES;
14864c8945a0SNathan Whitehorn return;
14874c8945a0SNathan Whitehorn }
14884c8945a0SNathan Whitehorn
14894c8945a0SNathan Whitehorn while (!feof(fd)) {
1490a96ef450SBaptiste Daroussin int ch;
1491a96ef450SBaptiste Daroussin long offset;
1492a96ef450SBaptiste Daroussin
1493f4f33ea0SBaptiste Daroussin if (ferror(fd))
1494f4f33ea0SBaptiste Daroussin break;
1495a96ef450SBaptiste Daroussin
14964c8945a0SNathan Whitehorn offset = 0;
1497f4f33ea0SBaptiste Daroussin while (((ch = getc(fd)) != '\n') && !feof(fd)) {
1498f4f33ea0SBaptiste Daroussin if ((ch == TAB) && (dialog_vars.tab_correct)) {
14994c8945a0SNathan Whitehorn offset += dialog_state.tab_len - (offset % dialog_state.tab_len);
1500f4f33ea0SBaptiste Daroussin } else {
15014c8945a0SNathan Whitehorn offset++;
1502f4f33ea0SBaptiste Daroussin }
1503f4f33ea0SBaptiste Daroussin }
15044c8945a0SNathan Whitehorn
15054c8945a0SNathan Whitehorn if (offset > len)
15067a1c0d96SNathan Whitehorn len = (int) offset;
15074c8945a0SNathan Whitehorn
15084c8945a0SNathan Whitehorn count++;
15094c8945a0SNathan Whitehorn }
15104c8945a0SNathan Whitehorn
15117a1c0d96SNathan Whitehorn /* now 'count' has the number of lines of fd and 'len' the max length */
15124c8945a0SNathan Whitehorn
15134c8945a0SNathan Whitehorn *height = MIN(SLINES, count + numlines + boxlines);
15144c8945a0SNathan Whitehorn *width = MIN(SCOLS, MAX((len + nc), mincols));
15154c8945a0SNathan Whitehorn /* here width and height can be maximized if > SCOLS|SLINES because
15164c8945a0SNathan Whitehorn textbox-like widgets don't put all <file> on the screen.
15174c8945a0SNathan Whitehorn Msgbox-like widget instead have to put all <text> correctly. */
15184c8945a0SNathan Whitehorn
15194c8945a0SNathan Whitehorn (void) fclose(fd);
15204c8945a0SNathan Whitehorn }
15214c8945a0SNathan Whitehorn
15224c8945a0SNathan Whitehorn /*
15234c8945a0SNathan Whitehorn * Draw a rectangular box with line drawing characters.
15244c8945a0SNathan Whitehorn *
15254c8945a0SNathan Whitehorn * borderchar is used to color the upper/left edges.
15264c8945a0SNathan Whitehorn *
15274c8945a0SNathan Whitehorn * boxchar is used to color the right/lower edges. It also is fill-color used
15284c8945a0SNathan Whitehorn * for the box contents.
15294c8945a0SNathan Whitehorn *
15304c8945a0SNathan Whitehorn * Normally, if you are drawing a scrollable box, use menubox_border_attr for
15314c8945a0SNathan Whitehorn * boxchar, and menubox_attr for borderchar since the scroll-arrows are drawn
15324c8945a0SNathan Whitehorn * with menubox_attr at the top, and menubox_border_attr at the bottom. That
15334c8945a0SNathan Whitehorn * also (given the default color choices) produces a recessed effect.
15344c8945a0SNathan Whitehorn *
15354c8945a0SNathan Whitehorn * If you want a raised effect (and are not going to use the scroll-arrows),
15364c8945a0SNathan Whitehorn * reverse this choice.
15374c8945a0SNathan Whitehorn */
15384c8945a0SNathan Whitehorn void
dlg_draw_box2(WINDOW * win,int y,int x,int height,int width,chtype boxchar,chtype borderchar,chtype borderchar2)15392a3e3873SBaptiste Daroussin dlg_draw_box2(WINDOW *win, int y, int x, int height, int width,
15402a3e3873SBaptiste Daroussin chtype boxchar, chtype borderchar, chtype borderchar2)
15414c8945a0SNathan Whitehorn {
15424c8945a0SNathan Whitehorn int i, j;
15437a1c0d96SNathan Whitehorn chtype save = dlg_get_attrs(win);
15444c8945a0SNathan Whitehorn
1545f4f33ea0SBaptiste Daroussin dlg_attrset(win, 0);
15464c8945a0SNathan Whitehorn for (i = 0; i < height; i++) {
15474c8945a0SNathan Whitehorn (void) wmove(win, y + i, x);
15484c8945a0SNathan Whitehorn for (j = 0; j < width; j++)
15494c8945a0SNathan Whitehorn if (!i && !j)
15504c8945a0SNathan Whitehorn (void) waddch(win, borderchar | dlg_boxchar(ACS_ULCORNER));
15514c8945a0SNathan Whitehorn else if (i == height - 1 && !j)
15524c8945a0SNathan Whitehorn (void) waddch(win, borderchar | dlg_boxchar(ACS_LLCORNER));
15534c8945a0SNathan Whitehorn else if (!i && j == width - 1)
15542a3e3873SBaptiste Daroussin (void) waddch(win, borderchar2 | dlg_boxchar(ACS_URCORNER));
15554c8945a0SNathan Whitehorn else if (i == height - 1 && j == width - 1)
15562a3e3873SBaptiste Daroussin (void) waddch(win, borderchar2 | dlg_boxchar(ACS_LRCORNER));
15574c8945a0SNathan Whitehorn else if (!i)
15584c8945a0SNathan Whitehorn (void) waddch(win, borderchar | dlg_boxchar(ACS_HLINE));
15594c8945a0SNathan Whitehorn else if (i == height - 1)
15602a3e3873SBaptiste Daroussin (void) waddch(win, borderchar2 | dlg_boxchar(ACS_HLINE));
15614c8945a0SNathan Whitehorn else if (!j)
15624c8945a0SNathan Whitehorn (void) waddch(win, borderchar | dlg_boxchar(ACS_VLINE));
15634c8945a0SNathan Whitehorn else if (j == width - 1)
15642a3e3873SBaptiste Daroussin (void) waddch(win, borderchar2 | dlg_boxchar(ACS_VLINE));
15654c8945a0SNathan Whitehorn else
15664c8945a0SNathan Whitehorn (void) waddch(win, boxchar | ' ');
15674c8945a0SNathan Whitehorn }
1568f4f33ea0SBaptiste Daroussin dlg_attrset(win, save);
15692a3e3873SBaptiste Daroussin }
15702a3e3873SBaptiste Daroussin
15712a3e3873SBaptiste Daroussin void
dlg_draw_box(WINDOW * win,int y,int x,int height,int width,chtype boxchar,chtype borderchar)15722a3e3873SBaptiste Daroussin dlg_draw_box(WINDOW *win, int y, int x, int height, int width,
15732a3e3873SBaptiste Daroussin chtype boxchar, chtype borderchar)
15742a3e3873SBaptiste Daroussin {
15752a3e3873SBaptiste Daroussin dlg_draw_box2(win, y, x, height, width, boxchar, borderchar, boxchar);
15764c8945a0SNathan Whitehorn }
15774c8945a0SNathan Whitehorn
1578a96ef450SBaptiste Daroussin /*
1579a96ef450SBaptiste Daroussin * Search the given 'list' for the given window 'win'. Typically 'win' is an
1580a96ef450SBaptiste Daroussin * input-window, i.e., a window where we might use wgetch.
1581a96ef450SBaptiste Daroussin *
1582a96ef450SBaptiste Daroussin * The all-windows list has normal- and shadow-windows. Since we never use the
1583a96ef450SBaptiste Daroussin * shadow as an input window, normally we just look for the normal-window.
1584a96ef450SBaptiste Daroussin *
1585a96ef450SBaptiste Daroussin * However, the all-subwindows list stores parent/child windows rather than
1586a96ef450SBaptiste Daroussin * normal/shadow windows. When searching that list, we look for the child
1587a96ef450SBaptiste Daroussin * window (in the .shadow field).
1588a96ef450SBaptiste Daroussin */
1589682c9e0fSNathan Whitehorn static DIALOG_WINDOWS *
find_window(DIALOG_WINDOWS * list,WINDOW * win,bool normal)1590a96ef450SBaptiste Daroussin find_window(DIALOG_WINDOWS * list, WINDOW *win, bool normal)
1591682c9e0fSNathan Whitehorn {
1592682c9e0fSNathan Whitehorn DIALOG_WINDOWS *result = 0;
1593682c9e0fSNathan Whitehorn DIALOG_WINDOWS *p;
1594682c9e0fSNathan Whitehorn
1595a96ef450SBaptiste Daroussin for (p = list; p != 0; p = p->next) {
1596a96ef450SBaptiste Daroussin WINDOW *check = normal ? p->normal : p->shadow;
1597a96ef450SBaptiste Daroussin if (check == win) {
1598682c9e0fSNathan Whitehorn result = p;
1599682c9e0fSNathan Whitehorn break;
1600682c9e0fSNathan Whitehorn }
1601682c9e0fSNathan Whitehorn }
1602682c9e0fSNathan Whitehorn return result;
1603682c9e0fSNathan Whitehorn }
1604682c9e0fSNathan Whitehorn
1605a96ef450SBaptiste Daroussin #define SearchTopWindows(win) find_window(dialog_state.all_windows, win, TRUE)
1606a96ef450SBaptiste Daroussin #define SearchSubWindows(win) find_window(dialog_state.all_subwindows, win, FALSE)
1607a96ef450SBaptiste Daroussin
1608a96ef450SBaptiste Daroussin /*
1609a96ef450SBaptiste Daroussin * Check for the existence of a window, e.g., when used for input or updating
1610a96ef450SBaptiste Daroussin * the display. This is used in dlg_getc() and related functions, to guard
1611a96ef450SBaptiste Daroussin * against an asynchronous window-deletion that might invalidate the input
1612a96ef450SBaptiste Daroussin * window used in dlg_getc().
1613a96ef450SBaptiste Daroussin */
1614a96ef450SBaptiste Daroussin DIALOG_WINDOWS *
_dlg_find_window(WINDOW * win)1615a96ef450SBaptiste Daroussin _dlg_find_window(WINDOW *win)
1616a96ef450SBaptiste Daroussin {
1617a96ef450SBaptiste Daroussin DIALOG_WINDOWS *result = 0;
1618a96ef450SBaptiste Daroussin
1619a96ef450SBaptiste Daroussin if ((result = SearchTopWindows(win)) == NULL)
1620a96ef450SBaptiste Daroussin result = SearchSubWindows(win);
1621a96ef450SBaptiste Daroussin return result;
1622a96ef450SBaptiste Daroussin }
1623a96ef450SBaptiste Daroussin
16244c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
16254c8945a0SNathan Whitehorn /*
1626682c9e0fSNathan Whitehorn * If we have wchgat(), use that for updating shadow attributes, to work with
1627682c9e0fSNathan Whitehorn * wide-character data.
1628682c9e0fSNathan Whitehorn */
1629682c9e0fSNathan Whitehorn
1630682c9e0fSNathan Whitehorn /*
1631682c9e0fSNathan Whitehorn * Check if the given point is "in" the given window. If so, return the window
1632682c9e0fSNathan Whitehorn * pointer, otherwise null.
1633682c9e0fSNathan Whitehorn */
1634682c9e0fSNathan Whitehorn static WINDOW *
in_window(WINDOW * win,int y,int x)1635682c9e0fSNathan Whitehorn in_window(WINDOW *win, int y, int x)
1636682c9e0fSNathan Whitehorn {
1637682c9e0fSNathan Whitehorn WINDOW *result = 0;
1638682c9e0fSNathan Whitehorn int y_base = getbegy(win);
1639682c9e0fSNathan Whitehorn int x_base = getbegx(win);
1640682c9e0fSNathan Whitehorn int y_last = getmaxy(win) + y_base;
1641682c9e0fSNathan Whitehorn int x_last = getmaxx(win) + x_base;
1642682c9e0fSNathan Whitehorn
1643682c9e0fSNathan Whitehorn if (y >= y_base && y <= y_last && x >= x_base && x <= x_last)
1644682c9e0fSNathan Whitehorn result = win;
1645682c9e0fSNathan Whitehorn return result;
1646682c9e0fSNathan Whitehorn }
1647682c9e0fSNathan Whitehorn
1648682c9e0fSNathan Whitehorn static WINDOW *
window_at_cell(DIALOG_WINDOWS * dw,int y,int x)1649682c9e0fSNathan Whitehorn window_at_cell(DIALOG_WINDOWS * dw, int y, int x)
1650682c9e0fSNathan Whitehorn {
1651682c9e0fSNathan Whitehorn WINDOW *result = 0;
1652682c9e0fSNathan Whitehorn DIALOG_WINDOWS *p;
1653682c9e0fSNathan Whitehorn int y_want = y + getbegy(dw->shadow);
1654682c9e0fSNathan Whitehorn int x_want = x + getbegx(dw->shadow);
1655682c9e0fSNathan Whitehorn
1656682c9e0fSNathan Whitehorn for (p = dialog_state.all_windows; p != 0; p = p->next) {
1657682c9e0fSNathan Whitehorn if (dw->normal != p->normal
1658682c9e0fSNathan Whitehorn && dw->shadow != p->normal
1659682c9e0fSNathan Whitehorn && (result = in_window(p->normal, y_want, x_want)) != 0) {
1660682c9e0fSNathan Whitehorn break;
1661682c9e0fSNathan Whitehorn }
1662682c9e0fSNathan Whitehorn }
1663682c9e0fSNathan Whitehorn if (result == 0) {
1664682c9e0fSNathan Whitehorn result = stdscr;
1665682c9e0fSNathan Whitehorn }
1666682c9e0fSNathan Whitehorn return result;
1667682c9e0fSNathan Whitehorn }
1668682c9e0fSNathan Whitehorn
1669682c9e0fSNathan Whitehorn static bool
in_shadow(WINDOW * normal,WINDOW * shadow,int y,int x)1670682c9e0fSNathan Whitehorn in_shadow(WINDOW *normal, WINDOW *shadow, int y, int x)
1671682c9e0fSNathan Whitehorn {
1672682c9e0fSNathan Whitehorn bool result = FALSE;
1673682c9e0fSNathan Whitehorn int ybase = getbegy(normal);
1674682c9e0fSNathan Whitehorn int ylast = getmaxy(normal) + ybase;
1675682c9e0fSNathan Whitehorn int xbase = getbegx(normal);
1676682c9e0fSNathan Whitehorn int xlast = getmaxx(normal) + xbase;
1677682c9e0fSNathan Whitehorn
1678682c9e0fSNathan Whitehorn y += getbegy(shadow);
1679682c9e0fSNathan Whitehorn x += getbegx(shadow);
1680682c9e0fSNathan Whitehorn
1681682c9e0fSNathan Whitehorn if (y >= ybase + SHADOW_ROWS
1682682c9e0fSNathan Whitehorn && y < ylast + SHADOW_ROWS
1683682c9e0fSNathan Whitehorn && x >= xlast
1684682c9e0fSNathan Whitehorn && x < xlast + SHADOW_COLS) {
1685682c9e0fSNathan Whitehorn /* in the right-side */
1686682c9e0fSNathan Whitehorn result = TRUE;
1687682c9e0fSNathan Whitehorn } else if (y >= ylast
1688682c9e0fSNathan Whitehorn && y < ylast + SHADOW_ROWS
1689682c9e0fSNathan Whitehorn && x >= ybase + SHADOW_COLS
1690682c9e0fSNathan Whitehorn && x < ylast + SHADOW_COLS) {
1691682c9e0fSNathan Whitehorn /* check the bottom */
1692682c9e0fSNathan Whitehorn result = TRUE;
1693682c9e0fSNathan Whitehorn }
1694682c9e0fSNathan Whitehorn
1695682c9e0fSNathan Whitehorn return result;
1696682c9e0fSNathan Whitehorn }
1697682c9e0fSNathan Whitehorn
1698682c9e0fSNathan Whitehorn /*
1699682c9e0fSNathan Whitehorn * When erasing a shadow, check each cell to make sure that it is not part of
1700682c9e0fSNathan Whitehorn * another box's shadow. This is a little complicated since most shadows are
1701682c9e0fSNathan Whitehorn * merged onto stdscr.
1702682c9e0fSNathan Whitehorn */
1703682c9e0fSNathan Whitehorn static bool
last_shadow(DIALOG_WINDOWS * dw,int y,int x)1704682c9e0fSNathan Whitehorn last_shadow(DIALOG_WINDOWS * dw, int y, int x)
1705682c9e0fSNathan Whitehorn {
1706682c9e0fSNathan Whitehorn DIALOG_WINDOWS *p;
1707682c9e0fSNathan Whitehorn bool result = TRUE;
1708682c9e0fSNathan Whitehorn
1709682c9e0fSNathan Whitehorn for (p = dialog_state.all_windows; p != 0; p = p->next) {
1710682c9e0fSNathan Whitehorn if (p->normal != dw->normal
1711682c9e0fSNathan Whitehorn && in_shadow(p->normal, dw->shadow, y, x)) {
1712682c9e0fSNathan Whitehorn result = FALSE;
1713682c9e0fSNathan Whitehorn break;
1714682c9e0fSNathan Whitehorn }
1715682c9e0fSNathan Whitehorn }
1716682c9e0fSNathan Whitehorn return result;
1717682c9e0fSNathan Whitehorn }
1718682c9e0fSNathan Whitehorn
1719682c9e0fSNathan Whitehorn static void
repaint_cell(DIALOG_WINDOWS * dw,bool draw,int y,int x)1720682c9e0fSNathan Whitehorn repaint_cell(DIALOG_WINDOWS * dw, bool draw, int y, int x)
1721682c9e0fSNathan Whitehorn {
1722682c9e0fSNathan Whitehorn WINDOW *win = dw->shadow;
1723682c9e0fSNathan Whitehorn WINDOW *cellwin;
1724682c9e0fSNathan Whitehorn int y2, x2;
1725682c9e0fSNathan Whitehorn
1726682c9e0fSNathan Whitehorn if ((cellwin = window_at_cell(dw, y, x)) != 0
1727682c9e0fSNathan Whitehorn && (draw || last_shadow(dw, y, x))
1728682c9e0fSNathan Whitehorn && (y2 = (y + getbegy(win) - getbegy(cellwin))) >= 0
1729682c9e0fSNathan Whitehorn && (x2 = (x + getbegx(win) - getbegx(cellwin))) >= 0
1730682c9e0fSNathan Whitehorn && wmove(cellwin, y2, x2) != ERR) {
1731682c9e0fSNathan Whitehorn chtype the_cell = dlg_get_attrs(cellwin);
1732682c9e0fSNathan Whitehorn chtype the_attr = (draw ? shadow_attr : the_cell);
1733682c9e0fSNathan Whitehorn
1734f4f33ea0SBaptiste Daroussin if (winch(cellwin) & A_ALTCHARSET) {
1735682c9e0fSNathan Whitehorn the_attr |= A_ALTCHARSET;
1736682c9e0fSNathan Whitehorn }
1737682c9e0fSNathan Whitehorn #if USE_WCHGAT
1738682c9e0fSNathan Whitehorn wchgat(cellwin, 1,
1739682c9e0fSNathan Whitehorn the_attr & (chtype) (~A_COLOR),
17402a3e3873SBaptiste Daroussin (short) PAIR_NUMBER(the_attr),
1741682c9e0fSNathan Whitehorn NULL);
1742682c9e0fSNathan Whitehorn #else
1743682c9e0fSNathan Whitehorn {
1744682c9e0fSNathan Whitehorn chtype the_char = ((winch(cellwin) & A_CHARTEXT) | the_attr);
1745682c9e0fSNathan Whitehorn (void) waddch(cellwin, the_char);
1746682c9e0fSNathan Whitehorn }
1747682c9e0fSNathan Whitehorn #endif
1748682c9e0fSNathan Whitehorn wnoutrefresh(cellwin);
1749682c9e0fSNathan Whitehorn }
1750682c9e0fSNathan Whitehorn }
1751682c9e0fSNathan Whitehorn
1752682c9e0fSNathan Whitehorn #define RepaintCell(dw, draw, y, x) repaint_cell(dw, draw, y, x)
1753682c9e0fSNathan Whitehorn
1754682c9e0fSNathan Whitehorn static void
repaint_shadow(DIALOG_WINDOWS * dw,bool draw,int y,int x,int height,int width)1755682c9e0fSNathan Whitehorn repaint_shadow(DIALOG_WINDOWS * dw, bool draw, int y, int x, int height, int width)
1756682c9e0fSNathan Whitehorn {
1757a96ef450SBaptiste Daroussin if (UseShadow(dw)) {
1758682c9e0fSNathan Whitehorn int i, j;
1759682c9e0fSNathan Whitehorn
1760682c9e0fSNathan Whitehorn #if !USE_WCHGAT
1761682c9e0fSNathan Whitehorn chtype save = dlg_get_attrs(dw->shadow);
1762f4f33ea0SBaptiste Daroussin dlg_attrset(dw->shadow, draw ? shadow_attr : screen_attr);
1763682c9e0fSNathan Whitehorn #endif
1764682c9e0fSNathan Whitehorn for (i = 0; i < SHADOW_ROWS; ++i) {
1765682c9e0fSNathan Whitehorn for (j = 0; j < width; ++j) {
1766682c9e0fSNathan Whitehorn RepaintCell(dw, draw, i + y + height, j + x + SHADOW_COLS);
1767682c9e0fSNathan Whitehorn }
1768682c9e0fSNathan Whitehorn }
1769682c9e0fSNathan Whitehorn for (i = 0; i < height; i++) {
1770682c9e0fSNathan Whitehorn for (j = 0; j < SHADOW_COLS; ++j) {
1771682c9e0fSNathan Whitehorn RepaintCell(dw, draw, i + y + SHADOW_ROWS, j + x + width);
1772682c9e0fSNathan Whitehorn }
1773682c9e0fSNathan Whitehorn }
1774682c9e0fSNathan Whitehorn (void) wnoutrefresh(dw->shadow);
1775682c9e0fSNathan Whitehorn #if !USE_WCHGAT
1776f4f33ea0SBaptiste Daroussin dlg_attrset(dw->shadow, save);
1777682c9e0fSNathan Whitehorn #endif
1778682c9e0fSNathan Whitehorn }
1779682c9e0fSNathan Whitehorn }
1780682c9e0fSNathan Whitehorn
1781682c9e0fSNathan Whitehorn /*
17824c8945a0SNathan Whitehorn * Draw a shadow on the parent window corresponding to the right- and
17834c8945a0SNathan Whitehorn * bottom-edge of the child window, to give a 3-dimensional look.
17844c8945a0SNathan Whitehorn */
17854c8945a0SNathan Whitehorn static void
draw_childs_shadow(DIALOG_WINDOWS * dw)1786682c9e0fSNathan Whitehorn draw_childs_shadow(DIALOG_WINDOWS * dw)
17874c8945a0SNathan Whitehorn {
1788682c9e0fSNathan Whitehorn if (UseShadow(dw)) {
1789682c9e0fSNathan Whitehorn repaint_shadow(dw,
1790682c9e0fSNathan Whitehorn TRUE,
1791682c9e0fSNathan Whitehorn getbegy(dw->normal) - getbegy(dw->shadow),
1792682c9e0fSNathan Whitehorn getbegx(dw->normal) - getbegx(dw->shadow),
1793682c9e0fSNathan Whitehorn getmaxy(dw->normal),
1794682c9e0fSNathan Whitehorn getmaxx(dw->normal));
1795682c9e0fSNathan Whitehorn }
1796682c9e0fSNathan Whitehorn }
17974c8945a0SNathan Whitehorn
1798682c9e0fSNathan Whitehorn /*
1799682c9e0fSNathan Whitehorn * Erase a shadow on the parent window corresponding to the right- and
1800682c9e0fSNathan Whitehorn * bottom-edge of the child window.
1801682c9e0fSNathan Whitehorn */
1802682c9e0fSNathan Whitehorn static void
erase_childs_shadow(DIALOG_WINDOWS * dw)1803682c9e0fSNathan Whitehorn erase_childs_shadow(DIALOG_WINDOWS * dw)
1804682c9e0fSNathan Whitehorn {
1805682c9e0fSNathan Whitehorn if (UseShadow(dw)) {
1806682c9e0fSNathan Whitehorn repaint_shadow(dw,
1807682c9e0fSNathan Whitehorn FALSE,
1808682c9e0fSNathan Whitehorn getbegy(dw->normal) - getbegy(dw->shadow),
1809682c9e0fSNathan Whitehorn getbegx(dw->normal) - getbegx(dw->shadow),
1810682c9e0fSNathan Whitehorn getmaxy(dw->normal),
1811682c9e0fSNathan Whitehorn getmaxx(dw->normal));
18124c8945a0SNathan Whitehorn }
18134c8945a0SNathan Whitehorn }
18144c8945a0SNathan Whitehorn
18154c8945a0SNathan Whitehorn /*
18164c8945a0SNathan Whitehorn * Draw shadows along the right and bottom edge to give a more 3D look
1817682c9e0fSNathan Whitehorn * to the boxes.
18184c8945a0SNathan Whitehorn */
18194c8945a0SNathan Whitehorn void
dlg_draw_shadow(WINDOW * win,int y,int x,int height,int width)18204c8945a0SNathan Whitehorn dlg_draw_shadow(WINDOW *win, int y, int x, int height, int width)
18214c8945a0SNathan Whitehorn {
1822a96ef450SBaptiste Daroussin repaint_shadow(SearchTopWindows(win), TRUE, y, x, height, width);
18234c8945a0SNathan Whitehorn }
18244c8945a0SNathan Whitehorn #endif /* HAVE_COLOR */
18254c8945a0SNathan Whitehorn
18264c8945a0SNathan Whitehorn /*
18274c8945a0SNathan Whitehorn * Allow shell scripts to remap the exit codes so they can distinguish ESC
18284c8945a0SNathan Whitehorn * from ERROR.
18294c8945a0SNathan Whitehorn */
18304c8945a0SNathan Whitehorn void
dlg_exit(int code)18314c8945a0SNathan Whitehorn dlg_exit(int code)
18324c8945a0SNathan Whitehorn {
18334c8945a0SNathan Whitehorn /* *INDENT-OFF* */
18344c8945a0SNathan Whitehorn static const struct {
18354c8945a0SNathan Whitehorn int code;
18364c8945a0SNathan Whitehorn const char *name;
18374c8945a0SNathan Whitehorn } table[] = {
18384c8945a0SNathan Whitehorn { DLG_EXIT_CANCEL, "DIALOG_CANCEL" },
18394c8945a0SNathan Whitehorn { DLG_EXIT_ERROR, "DIALOG_ERROR" },
18404c8945a0SNathan Whitehorn { DLG_EXIT_ESC, "DIALOG_ESC" },
18414c8945a0SNathan Whitehorn { DLG_EXIT_EXTRA, "DIALOG_EXTRA" },
18424c8945a0SNathan Whitehorn { DLG_EXIT_HELP, "DIALOG_HELP" },
18434c8945a0SNathan Whitehorn { DLG_EXIT_OK, "DIALOG_OK" },
18444c8945a0SNathan Whitehorn { DLG_EXIT_ITEM_HELP, "DIALOG_ITEM_HELP" },
1845a96ef450SBaptiste Daroussin { DLG_EXIT_TIMEOUT, "DIALOG_TIMEOUT" },
18464c8945a0SNathan Whitehorn };
18474c8945a0SNathan Whitehorn /* *INDENT-ON* */
18484c8945a0SNathan Whitehorn
18494c8945a0SNathan Whitehorn unsigned n;
18504c8945a0SNathan Whitehorn bool overridden = FALSE;
18514c8945a0SNathan Whitehorn
18524c8945a0SNathan Whitehorn retry:
1853a96ef450SBaptiste Daroussin for (n = 0; n < TableSize(table); n++) {
18544c8945a0SNathan Whitehorn if (table[n].code == code) {
1855a96ef450SBaptiste Daroussin if (dlg_getenv_num(table[n].name, &code)) {
18564c8945a0SNathan Whitehorn overridden = TRUE;
18574c8945a0SNathan Whitehorn }
18584c8945a0SNathan Whitehorn break;
18594c8945a0SNathan Whitehorn }
18604c8945a0SNathan Whitehorn }
18614c8945a0SNathan Whitehorn
18624c8945a0SNathan Whitehorn /*
18634c8945a0SNathan Whitehorn * Prior to 2004/12/19, a widget using --item-help would exit with "OK"
18644c8945a0SNathan Whitehorn * if the help button were selected. Now we want to exit with "HELP",
18654c8945a0SNathan Whitehorn * but allow the environment variable to override.
18664c8945a0SNathan Whitehorn */
18674c8945a0SNathan Whitehorn if (code == DLG_EXIT_ITEM_HELP && !overridden) {
18684c8945a0SNathan Whitehorn code = DLG_EXIT_HELP;
18694c8945a0SNathan Whitehorn goto retry;
18704c8945a0SNathan Whitehorn }
1871682c9e0fSNathan Whitehorn #ifdef HAVE_DLG_TRACE
1872682c9e0fSNathan Whitehorn dlg_trace((const char *) 0); /* close it */
1873682c9e0fSNathan Whitehorn #endif
1874682c9e0fSNathan Whitehorn
18754c8945a0SNathan Whitehorn #ifdef NO_LEAKS
18764c8945a0SNathan Whitehorn _dlg_inputstr_leaks();
1877a96ef450SBaptiste Daroussin #if defined(NCURSES_VERSION) && (defined(HAVE_CURSES_EXIT) || defined(HAVE__NC_FREE_AND_EXIT))
1878a96ef450SBaptiste Daroussin curses_exit(code);
18794c8945a0SNathan Whitehorn #endif
18804c8945a0SNathan Whitehorn #endif
18814c8945a0SNathan Whitehorn
18824c8945a0SNathan Whitehorn if (dialog_state.input == stdin) {
18834c8945a0SNathan Whitehorn exit(code);
18844c8945a0SNathan Whitehorn } else {
18854c8945a0SNathan Whitehorn /*
18864c8945a0SNathan Whitehorn * Just in case of using --input-fd option, do not
18874c8945a0SNathan Whitehorn * call atexit functions of ncurses which may hang.
18884c8945a0SNathan Whitehorn */
18894c8945a0SNathan Whitehorn if (dialog_state.input) {
18904c8945a0SNathan Whitehorn fclose(dialog_state.input);
18914c8945a0SNathan Whitehorn dialog_state.input = 0;
18924c8945a0SNathan Whitehorn }
18934c8945a0SNathan Whitehorn if (dialog_state.pipe_input) {
18944c8945a0SNathan Whitehorn if (dialog_state.pipe_input != stdin) {
18954c8945a0SNathan Whitehorn fclose(dialog_state.pipe_input);
18964c8945a0SNathan Whitehorn dialog_state.pipe_input = 0;
18974c8945a0SNathan Whitehorn }
18984c8945a0SNathan Whitehorn }
18994c8945a0SNathan Whitehorn _exit(code);
19004c8945a0SNathan Whitehorn }
19014c8945a0SNathan Whitehorn }
19024c8945a0SNathan Whitehorn
1903a96ef450SBaptiste Daroussin #define DATA(name) { DLG_EXIT_ ## name, #name }
1904a96ef450SBaptiste Daroussin /* *INDENT-OFF* */
1905a96ef450SBaptiste Daroussin static struct {
1906a96ef450SBaptiste Daroussin int code;
1907a96ef450SBaptiste Daroussin const char *name;
1908a96ef450SBaptiste Daroussin } exit_codenames[] = {
1909a96ef450SBaptiste Daroussin DATA(ESC),
1910a96ef450SBaptiste Daroussin DATA(UNKNOWN),
1911a96ef450SBaptiste Daroussin DATA(ERROR),
1912a96ef450SBaptiste Daroussin DATA(OK),
1913a96ef450SBaptiste Daroussin DATA(CANCEL),
1914a96ef450SBaptiste Daroussin DATA(HELP),
1915a96ef450SBaptiste Daroussin DATA(EXTRA),
1916a96ef450SBaptiste Daroussin DATA(ITEM_HELP),
1917a96ef450SBaptiste Daroussin };
1918a96ef450SBaptiste Daroussin #undef DATA
1919a96ef450SBaptiste Daroussin /* *INDENT-ON* */
1920a96ef450SBaptiste Daroussin
1921a96ef450SBaptiste Daroussin const char *
dlg_exitcode2s(int code)1922a96ef450SBaptiste Daroussin dlg_exitcode2s(int code)
1923a96ef450SBaptiste Daroussin {
1924a96ef450SBaptiste Daroussin const char *result = "?";
1925a96ef450SBaptiste Daroussin size_t n;
1926a96ef450SBaptiste Daroussin
1927a96ef450SBaptiste Daroussin for (n = 0; n < TableSize(exit_codenames); ++n) {
1928a96ef450SBaptiste Daroussin if (exit_codenames[n].code == code) {
1929a96ef450SBaptiste Daroussin result = exit_codenames[n].name;
1930a96ef450SBaptiste Daroussin break;
1931a96ef450SBaptiste Daroussin }
1932a96ef450SBaptiste Daroussin }
1933a96ef450SBaptiste Daroussin return result;
1934a96ef450SBaptiste Daroussin }
1935a96ef450SBaptiste Daroussin
1936a96ef450SBaptiste Daroussin int
dlg_exitname2n(const char * name)1937a96ef450SBaptiste Daroussin dlg_exitname2n(const char *name)
1938a96ef450SBaptiste Daroussin {
1939a96ef450SBaptiste Daroussin int result = DLG_EXIT_UNKNOWN;
1940a96ef450SBaptiste Daroussin size_t n;
1941a96ef450SBaptiste Daroussin
1942a96ef450SBaptiste Daroussin for (n = 0; n < TableSize(exit_codenames); ++n) {
1943a96ef450SBaptiste Daroussin if (!dlg_strcmp(exit_codenames[n].name, name)) {
1944a96ef450SBaptiste Daroussin result = exit_codenames[n].code;
1945a96ef450SBaptiste Daroussin break;
1946a96ef450SBaptiste Daroussin }
1947a96ef450SBaptiste Daroussin }
1948a96ef450SBaptiste Daroussin return result;
1949a96ef450SBaptiste Daroussin }
1950a96ef450SBaptiste Daroussin
19514c8945a0SNathan Whitehorn /* quit program killing all tailbg */
19524c8945a0SNathan Whitehorn void
dlg_exiterr(const char * fmt,...)19534c8945a0SNathan Whitehorn dlg_exiterr(const char *fmt, ...)
19544c8945a0SNathan Whitehorn {
19554c8945a0SNathan Whitehorn int retval;
19564c8945a0SNathan Whitehorn va_list ap;
19574c8945a0SNathan Whitehorn
19584c8945a0SNathan Whitehorn end_dialog();
19594c8945a0SNathan Whitehorn
19604c8945a0SNathan Whitehorn (void) fputc('\n', stderr);
19614c8945a0SNathan Whitehorn va_start(ap, fmt);
19624c8945a0SNathan Whitehorn (void) vfprintf(stderr, fmt, ap);
19634c8945a0SNathan Whitehorn va_end(ap);
19644c8945a0SNathan Whitehorn (void) fputc('\n', stderr);
19654c8945a0SNathan Whitehorn
1966a96ef450SBaptiste Daroussin #ifdef HAVE_DLG_TRACE
1967a96ef450SBaptiste Daroussin va_start(ap, fmt);
1968a96ef450SBaptiste Daroussin dlg_trace_msg("## Error: ");
1969a96ef450SBaptiste Daroussin dlg_trace_va_msg(fmt, ap);
1970a96ef450SBaptiste Daroussin va_end(ap);
1971a96ef450SBaptiste Daroussin #endif
1972a96ef450SBaptiste Daroussin
19734c8945a0SNathan Whitehorn dlg_killall_bg(&retval);
19744c8945a0SNathan Whitehorn
19754c8945a0SNathan Whitehorn (void) fflush(stderr);
19764c8945a0SNathan Whitehorn (void) fflush(stdout);
1977a96ef450SBaptiste Daroussin dlg_exit(strcmp(fmt, "timeout") == 0 ? DLG_EXIT_TIMEOUT : DLG_EXIT_ERROR);
1978a96ef450SBaptiste Daroussin }
1979a96ef450SBaptiste Daroussin
1980a96ef450SBaptiste Daroussin /*
1981a96ef450SBaptiste Daroussin * Get a string from the environment, rejecting those which are entirely blank.
1982a96ef450SBaptiste Daroussin */
1983a96ef450SBaptiste Daroussin char *
dlg_getenv_str(const char * name)1984a96ef450SBaptiste Daroussin dlg_getenv_str(const char *name)
1985a96ef450SBaptiste Daroussin {
1986a96ef450SBaptiste Daroussin char *result = getenv(name);
1987a96ef450SBaptiste Daroussin if (result != NULL) {
1988a96ef450SBaptiste Daroussin while (*result != '\0' && isspace(UCH(*result)))
1989a96ef450SBaptiste Daroussin ++result;
1990a96ef450SBaptiste Daroussin if (*result == '\0')
1991a96ef450SBaptiste Daroussin result = NULL;
1992a96ef450SBaptiste Daroussin }
1993a96ef450SBaptiste Daroussin return result;
1994a96ef450SBaptiste Daroussin }
1995a96ef450SBaptiste Daroussin
1996a96ef450SBaptiste Daroussin /*
1997a96ef450SBaptiste Daroussin * Get a number from the environment:
1998a96ef450SBaptiste Daroussin * + If the caller provides a pointer in the second parameter, return
1999a96ef450SBaptiste Daroussin * success/failure for the function return, and the actual value via the
2000a96ef450SBaptiste Daroussin * pointer. Use this for decoding arbitrary numbers, e.g., negative or zero.
2001a96ef450SBaptiste Daroussin * + If the caller does not provide a pointer, return the decoded value for
2002a96ef450SBaptiste Daroussin * the function-return. Use this when only values greater than zero are
2003a96ef450SBaptiste Daroussin * useful.
2004a96ef450SBaptiste Daroussin */
2005a96ef450SBaptiste Daroussin int
dlg_getenv_num(const char * name,int * value)2006a96ef450SBaptiste Daroussin dlg_getenv_num(const char *name, int *value)
2007a96ef450SBaptiste Daroussin {
2008a96ef450SBaptiste Daroussin int result = 0;
2009a96ef450SBaptiste Daroussin char *data = getenv(name);
2010a96ef450SBaptiste Daroussin if (data != NULL) {
2011a96ef450SBaptiste Daroussin char *temp = NULL;
2012a96ef450SBaptiste Daroussin long check = strtol(data, &temp, 0);
2013a96ef450SBaptiste Daroussin if (temp != 0 && temp != data && *temp == '\0') {
2014a96ef450SBaptiste Daroussin result = (int) check;
2015a96ef450SBaptiste Daroussin if (value != NULL) {
2016a96ef450SBaptiste Daroussin *value = result;
2017a96ef450SBaptiste Daroussin result = 1;
2018a96ef450SBaptiste Daroussin }
2019a96ef450SBaptiste Daroussin }
2020a96ef450SBaptiste Daroussin }
2021a96ef450SBaptiste Daroussin return result;
20224c8945a0SNathan Whitehorn }
20234c8945a0SNathan Whitehorn
20244c8945a0SNathan Whitehorn void
dlg_beeping(void)20254c8945a0SNathan Whitehorn dlg_beeping(void)
20264c8945a0SNathan Whitehorn {
20274c8945a0SNathan Whitehorn if (dialog_vars.beep_signal) {
20284c8945a0SNathan Whitehorn (void) beep();
20294c8945a0SNathan Whitehorn dialog_vars.beep_signal = 0;
20304c8945a0SNathan Whitehorn }
20314c8945a0SNathan Whitehorn }
20324c8945a0SNathan Whitehorn
20334c8945a0SNathan Whitehorn void
dlg_print_size(int height,int width)20344c8945a0SNathan Whitehorn dlg_print_size(int height, int width)
20354c8945a0SNathan Whitehorn {
2036f4f33ea0SBaptiste Daroussin if (dialog_vars.print_siz) {
20374c8945a0SNathan Whitehorn fprintf(dialog_state.output, "Size: %d, %d\n", height, width);
2038f4f33ea0SBaptiste Daroussin DLG_TRACE(("# print size: %dx%d\n", height, width));
2039f4f33ea0SBaptiste Daroussin }
20404c8945a0SNathan Whitehorn }
20414c8945a0SNathan Whitehorn
20424c8945a0SNathan Whitehorn void
dlg_ctl_size(int height,int width)20434c8945a0SNathan Whitehorn dlg_ctl_size(int height, int width)
20444c8945a0SNathan Whitehorn {
20454c8945a0SNathan Whitehorn if (dialog_vars.size_err) {
20464c8945a0SNathan Whitehorn if ((width > COLS) || (height > LINES)) {
20474c8945a0SNathan Whitehorn dlg_exiterr("Window too big. (height, width) = (%d, %d). Max allowed (%d, %d).",
20484c8945a0SNathan Whitehorn height, width, LINES, COLS);
20494c8945a0SNathan Whitehorn }
20504c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
20514c8945a0SNathan Whitehorn else if ((dialog_state.use_shadow)
20524c8945a0SNathan Whitehorn && ((width > SCOLS || height > SLINES))) {
20534c8945a0SNathan Whitehorn if ((width <= COLS) && (height <= LINES)) {
20544c8945a0SNathan Whitehorn /* try again, without shadows */
20554c8945a0SNathan Whitehorn dialog_state.use_shadow = 0;
20564c8945a0SNathan Whitehorn } else {
20574c8945a0SNathan Whitehorn dlg_exiterr("Window+Shadow too big. (height, width) = (%d, %d). Max allowed (%d, %d).",
20584c8945a0SNathan Whitehorn height, width, SLINES, SCOLS);
20594c8945a0SNathan Whitehorn }
20604c8945a0SNathan Whitehorn }
20614c8945a0SNathan Whitehorn #endif
20624c8945a0SNathan Whitehorn }
20634c8945a0SNathan Whitehorn }
20644c8945a0SNathan Whitehorn
20654c8945a0SNathan Whitehorn /*
20664c8945a0SNathan Whitehorn * If the --tab-correct was not selected, convert tabs to single spaces.
20674c8945a0SNathan Whitehorn */
20684c8945a0SNathan Whitehorn void
dlg_tab_correct_str(char * prompt)20694c8945a0SNathan Whitehorn dlg_tab_correct_str(char *prompt)
20704c8945a0SNathan Whitehorn {
20714c8945a0SNathan Whitehorn char *ptr;
20724c8945a0SNathan Whitehorn
20734c8945a0SNathan Whitehorn if (dialog_vars.tab_correct) {
20744c8945a0SNathan Whitehorn while ((ptr = strchr(prompt, TAB)) != NULL) {
20754c8945a0SNathan Whitehorn *ptr = ' ';
20764c8945a0SNathan Whitehorn prompt = ptr;
20774c8945a0SNathan Whitehorn }
20784c8945a0SNathan Whitehorn }
20794c8945a0SNathan Whitehorn }
20804c8945a0SNathan Whitehorn
20814c8945a0SNathan Whitehorn void
dlg_calc_listh(int * height,int * list_height,int item_no)20824c8945a0SNathan Whitehorn dlg_calc_listh(int *height, int *list_height, int item_no)
20834c8945a0SNathan Whitehorn {
20844c8945a0SNathan Whitehorn /* calculate new height and list_height */
20854c8945a0SNathan Whitehorn int rows = SLINES - (dialog_vars.begin_set ? dialog_vars.begin_y : 0);
20864c8945a0SNathan Whitehorn if (rows - (*height) > 0) {
20874c8945a0SNathan Whitehorn if (rows - (*height) > item_no)
20884c8945a0SNathan Whitehorn *list_height = item_no;
20894c8945a0SNathan Whitehorn else
20904c8945a0SNathan Whitehorn *list_height = rows - (*height);
20914c8945a0SNathan Whitehorn }
20924c8945a0SNathan Whitehorn (*height) += (*list_height);
20934c8945a0SNathan Whitehorn }
20944c8945a0SNathan Whitehorn
20954c8945a0SNathan Whitehorn /* obsolete */
20964c8945a0SNathan Whitehorn int
dlg_calc_listw(int item_no,char ** items,int group)20974c8945a0SNathan Whitehorn dlg_calc_listw(int item_no, char **items, int group)
20984c8945a0SNathan Whitehorn {
2099a96ef450SBaptiste Daroussin int i, len1 = 0, len2 = 0;
2100a96ef450SBaptiste Daroussin
21014c8945a0SNathan Whitehorn for (i = 0; i < (item_no * group); i += group) {
2102a96ef450SBaptiste Daroussin int n;
2103a96ef450SBaptiste Daroussin
21044c8945a0SNathan Whitehorn if ((n = dlg_count_columns(items[i])) > len1)
21054c8945a0SNathan Whitehorn len1 = n;
21064c8945a0SNathan Whitehorn if ((n = dlg_count_columns(items[i + 1])) > len2)
21074c8945a0SNathan Whitehorn len2 = n;
21084c8945a0SNathan Whitehorn }
21094c8945a0SNathan Whitehorn return len1 + len2;
21104c8945a0SNathan Whitehorn }
21114c8945a0SNathan Whitehorn
21124c8945a0SNathan Whitehorn int
dlg_calc_list_width(int item_no,DIALOG_LISTITEM * items)21134c8945a0SNathan Whitehorn dlg_calc_list_width(int item_no, DIALOG_LISTITEM * items)
21144c8945a0SNathan Whitehorn {
21154c8945a0SNathan Whitehorn int n, i, len1 = 0, len2 = 0;
21162a3e3873SBaptiste Daroussin int bits = ((dialog_vars.no_tags ? 1 : 0)
21172a3e3873SBaptiste Daroussin + (dialog_vars.no_items ? 2 : 0));
21182a3e3873SBaptiste Daroussin
21194c8945a0SNathan Whitehorn for (i = 0; i < item_no; ++i) {
21202a3e3873SBaptiste Daroussin switch (bits) {
21212a3e3873SBaptiste Daroussin case 0:
21222a3e3873SBaptiste Daroussin /* FALLTHRU */
21232a3e3873SBaptiste Daroussin case 1:
21244c8945a0SNathan Whitehorn if ((n = dlg_count_columns(items[i].name)) > len1)
21254c8945a0SNathan Whitehorn len1 = n;
21264c8945a0SNathan Whitehorn if ((n = dlg_count_columns(items[i].text)) > len2)
21274c8945a0SNathan Whitehorn len2 = n;
21282a3e3873SBaptiste Daroussin break;
21292a3e3873SBaptiste Daroussin case 2:
21302a3e3873SBaptiste Daroussin /* FALLTHRU */
21312a3e3873SBaptiste Daroussin case 3:
21322a3e3873SBaptiste Daroussin if ((n = dlg_count_columns(items[i].name)) > len1)
21332a3e3873SBaptiste Daroussin len1 = n;
21342a3e3873SBaptiste Daroussin break;
21352a3e3873SBaptiste Daroussin }
21364c8945a0SNathan Whitehorn }
21374c8945a0SNathan Whitehorn return len1 + len2;
21384c8945a0SNathan Whitehorn }
21394c8945a0SNathan Whitehorn
21404c8945a0SNathan Whitehorn char *
dlg_strempty(void)21414c8945a0SNathan Whitehorn dlg_strempty(void)
21424c8945a0SNathan Whitehorn {
21434c8945a0SNathan Whitehorn static char empty[] = "";
21444c8945a0SNathan Whitehorn return empty;
21454c8945a0SNathan Whitehorn }
21464c8945a0SNathan Whitehorn
21474c8945a0SNathan Whitehorn char *
dlg_strclone(const char * cprompt)21484c8945a0SNathan Whitehorn dlg_strclone(const char *cprompt)
21494c8945a0SNathan Whitehorn {
2150f4f33ea0SBaptiste Daroussin char *prompt = 0;
2151f4f33ea0SBaptiste Daroussin if (cprompt != 0) {
2152f4f33ea0SBaptiste Daroussin prompt = dlg_malloc(char, strlen(cprompt) + 1);
21534c8945a0SNathan Whitehorn assert_ptr(prompt, "dlg_strclone");
21544c8945a0SNathan Whitehorn strcpy(prompt, cprompt);
2155f4f33ea0SBaptiste Daroussin }
21564c8945a0SNathan Whitehorn return prompt;
21574c8945a0SNathan Whitehorn }
21584c8945a0SNathan Whitehorn
21594c8945a0SNathan Whitehorn chtype
dlg_asciibox(chtype ch)21604c8945a0SNathan Whitehorn dlg_asciibox(chtype ch)
21614c8945a0SNathan Whitehorn {
21624c8945a0SNathan Whitehorn chtype result = 0;
21634c8945a0SNathan Whitehorn
21644c8945a0SNathan Whitehorn if (ch == ACS_ULCORNER)
21654c8945a0SNathan Whitehorn result = '+';
21664c8945a0SNathan Whitehorn else if (ch == ACS_LLCORNER)
21674c8945a0SNathan Whitehorn result = '+';
21684c8945a0SNathan Whitehorn else if (ch == ACS_URCORNER)
21694c8945a0SNathan Whitehorn result = '+';
21704c8945a0SNathan Whitehorn else if (ch == ACS_LRCORNER)
21714c8945a0SNathan Whitehorn result = '+';
21724c8945a0SNathan Whitehorn else if (ch == ACS_HLINE)
21734c8945a0SNathan Whitehorn result = '-';
21744c8945a0SNathan Whitehorn else if (ch == ACS_VLINE)
21754c8945a0SNathan Whitehorn result = '|';
21764c8945a0SNathan Whitehorn else if (ch == ACS_LTEE)
21774c8945a0SNathan Whitehorn result = '+';
21784c8945a0SNathan Whitehorn else if (ch == ACS_RTEE)
21794c8945a0SNathan Whitehorn result = '+';
21804c8945a0SNathan Whitehorn else if (ch == ACS_UARROW)
21814c8945a0SNathan Whitehorn result = '^';
21824c8945a0SNathan Whitehorn else if (ch == ACS_DARROW)
21834c8945a0SNathan Whitehorn result = 'v';
21844c8945a0SNathan Whitehorn
21854c8945a0SNathan Whitehorn return result;
21864c8945a0SNathan Whitehorn }
21874c8945a0SNathan Whitehorn
21884c8945a0SNathan Whitehorn chtype
dlg_boxchar(chtype ch)21894c8945a0SNathan Whitehorn dlg_boxchar(chtype ch)
21904c8945a0SNathan Whitehorn {
21914c8945a0SNathan Whitehorn chtype result = dlg_asciibox(ch);
21924c8945a0SNathan Whitehorn
21934c8945a0SNathan Whitehorn if (result != 0) {
21944c8945a0SNathan Whitehorn if (dialog_vars.ascii_lines)
21954c8945a0SNathan Whitehorn ch = result;
21964c8945a0SNathan Whitehorn else if (dialog_vars.no_lines)
21974c8945a0SNathan Whitehorn ch = ' ';
21984c8945a0SNathan Whitehorn }
21994c8945a0SNathan Whitehorn return ch;
22004c8945a0SNathan Whitehorn }
22014c8945a0SNathan Whitehorn
22024c8945a0SNathan Whitehorn int
dlg_box_x_ordinate(int width)22034c8945a0SNathan Whitehorn dlg_box_x_ordinate(int width)
22044c8945a0SNathan Whitehorn {
22054c8945a0SNathan Whitehorn int x;
22064c8945a0SNathan Whitehorn
22074c8945a0SNathan Whitehorn if (dialog_vars.begin_set == 1) {
22084c8945a0SNathan Whitehorn x = dialog_vars.begin_x;
22094c8945a0SNathan Whitehorn } else {
22104c8945a0SNathan Whitehorn /* center dialog box on screen unless --begin-set */
22114c8945a0SNathan Whitehorn x = (SCOLS - width) / 2;
22124c8945a0SNathan Whitehorn }
22134c8945a0SNathan Whitehorn return x;
22144c8945a0SNathan Whitehorn }
22154c8945a0SNathan Whitehorn
22164c8945a0SNathan Whitehorn int
dlg_box_y_ordinate(int height)22174c8945a0SNathan Whitehorn dlg_box_y_ordinate(int height)
22184c8945a0SNathan Whitehorn {
22194c8945a0SNathan Whitehorn int y;
22204c8945a0SNathan Whitehorn
22214c8945a0SNathan Whitehorn if (dialog_vars.begin_set == 1) {
22224c8945a0SNathan Whitehorn y = dialog_vars.begin_y;
22234c8945a0SNathan Whitehorn } else {
22244c8945a0SNathan Whitehorn /* center dialog box on screen unless --begin-set */
22254c8945a0SNathan Whitehorn y = (SLINES - height) / 2;
22264c8945a0SNathan Whitehorn }
22274c8945a0SNathan Whitehorn return y;
22284c8945a0SNathan Whitehorn }
22294c8945a0SNathan Whitehorn
22304c8945a0SNathan Whitehorn void
dlg_draw_title(WINDOW * win,const char * title)22314c8945a0SNathan Whitehorn dlg_draw_title(WINDOW *win, const char *title)
22324c8945a0SNathan Whitehorn {
22334c8945a0SNathan Whitehorn if (title != NULL) {
22344c8945a0SNathan Whitehorn chtype attr = A_NORMAL;
22357a1c0d96SNathan Whitehorn chtype save = dlg_get_attrs(win);
22364c8945a0SNathan Whitehorn int x = centered(getmaxx(win), title);
22374c8945a0SNathan Whitehorn
2238f4f33ea0SBaptiste Daroussin dlg_attrset(win, title_attr);
22394c8945a0SNathan Whitehorn wmove(win, 0, x);
22404c8945a0SNathan Whitehorn dlg_print_text(win, title, getmaxx(win) - x, &attr);
2241f4f33ea0SBaptiste Daroussin dlg_attrset(win, save);
2242febdb468SDevin Teske dlg_finish_string(title);
22434c8945a0SNathan Whitehorn }
22444c8945a0SNathan Whitehorn }
22454c8945a0SNathan Whitehorn
22464c8945a0SNathan Whitehorn void
dlg_draw_bottom_box2(WINDOW * win,chtype on_left,chtype on_right,chtype on_inside)22472a3e3873SBaptiste Daroussin dlg_draw_bottom_box2(WINDOW *win, chtype on_left, chtype on_right, chtype on_inside)
22484c8945a0SNathan Whitehorn {
22494c8945a0SNathan Whitehorn int width = getmaxx(win);
22504c8945a0SNathan Whitehorn int height = getmaxy(win);
22514c8945a0SNathan Whitehorn int i;
22524c8945a0SNathan Whitehorn
2253f4f33ea0SBaptiste Daroussin dlg_attrset(win, on_left);
22544c8945a0SNathan Whitehorn (void) wmove(win, height - 3, 0);
22554c8945a0SNathan Whitehorn (void) waddch(win, dlg_boxchar(ACS_LTEE));
22564c8945a0SNathan Whitehorn for (i = 0; i < width - 2; i++)
22574c8945a0SNathan Whitehorn (void) waddch(win, dlg_boxchar(ACS_HLINE));
2258f4f33ea0SBaptiste Daroussin dlg_attrset(win, on_right);
22594c8945a0SNathan Whitehorn (void) waddch(win, dlg_boxchar(ACS_RTEE));
2260f4f33ea0SBaptiste Daroussin dlg_attrset(win, on_inside);
22614c8945a0SNathan Whitehorn (void) wmove(win, height - 2, 1);
22624c8945a0SNathan Whitehorn for (i = 0; i < width - 2; i++)
22634c8945a0SNathan Whitehorn (void) waddch(win, ' ');
22644c8945a0SNathan Whitehorn }
22654c8945a0SNathan Whitehorn
22662a3e3873SBaptiste Daroussin void
dlg_draw_bottom_box(WINDOW * win)22672a3e3873SBaptiste Daroussin dlg_draw_bottom_box(WINDOW *win)
22682a3e3873SBaptiste Daroussin {
22692a3e3873SBaptiste Daroussin dlg_draw_bottom_box2(win, border_attr, dialog_attr, dialog_attr);
22702a3e3873SBaptiste Daroussin }
22712a3e3873SBaptiste Daroussin
22724c8945a0SNathan Whitehorn /*
22734c8945a0SNathan Whitehorn * Remove a window, repainting everything else. This would be simpler if we
22744c8945a0SNathan Whitehorn * used the panel library, but that is not _always_ available.
22754c8945a0SNathan Whitehorn */
22764c8945a0SNathan Whitehorn void
dlg_del_window(WINDOW * win)22774c8945a0SNathan Whitehorn dlg_del_window(WINDOW *win)
22784c8945a0SNathan Whitehorn {
22794c8945a0SNathan Whitehorn DIALOG_WINDOWS *p, *q, *r;
22804c8945a0SNathan Whitehorn
22814c8945a0SNathan Whitehorn /*
22824c8945a0SNathan Whitehorn * If --keep-window was set, do not delete/repaint the windows.
22834c8945a0SNathan Whitehorn */
22844c8945a0SNathan Whitehorn if (dialog_vars.keep_window)
22854c8945a0SNathan Whitehorn return;
22864c8945a0SNathan Whitehorn
22874c8945a0SNathan Whitehorn /* Leave the main window untouched if there are no background windows.
22884c8945a0SNathan Whitehorn * We do this so the current window will not be cleared on exit, allowing
22894c8945a0SNathan Whitehorn * things like the infobox demo to run without flicker.
22904c8945a0SNathan Whitehorn */
22914c8945a0SNathan Whitehorn if (dialog_state.getc_callbacks != 0) {
22924c8945a0SNathan Whitehorn touchwin(stdscr);
22934c8945a0SNathan Whitehorn wnoutrefresh(stdscr);
22944c8945a0SNathan Whitehorn }
22954c8945a0SNathan Whitehorn
22964c8945a0SNathan Whitehorn for (p = dialog_state.all_windows, q = r = 0; p != 0; r = p, p = p->next) {
22974c8945a0SNathan Whitehorn if (p->normal == win) {
22984c8945a0SNathan Whitehorn q = p; /* found a match - should be only one */
22994c8945a0SNathan Whitehorn if (r == 0) {
23004c8945a0SNathan Whitehorn dialog_state.all_windows = p->next;
23014c8945a0SNathan Whitehorn } else {
23024c8945a0SNathan Whitehorn r->next = p->next;
23034c8945a0SNathan Whitehorn }
23044c8945a0SNathan Whitehorn } else {
23054c8945a0SNathan Whitehorn if (p->shadow != 0) {
23064c8945a0SNathan Whitehorn touchwin(p->shadow);
23074c8945a0SNathan Whitehorn wnoutrefresh(p->shadow);
23084c8945a0SNathan Whitehorn }
23094c8945a0SNathan Whitehorn touchwin(p->normal);
23104c8945a0SNathan Whitehorn wnoutrefresh(p->normal);
23114c8945a0SNathan Whitehorn }
23124c8945a0SNathan Whitehorn }
23134c8945a0SNathan Whitehorn
23144c8945a0SNathan Whitehorn if (q) {
2315682c9e0fSNathan Whitehorn if (dialog_state.all_windows != 0)
2316682c9e0fSNathan Whitehorn erase_childs_shadow(q);
23172a3e3873SBaptiste Daroussin del_subwindows(q->normal);
23184c8945a0SNathan Whitehorn dlg_unregister_window(q->normal);
23192a3e3873SBaptiste Daroussin delwin(q->normal);
23204c8945a0SNathan Whitehorn free(q);
23214c8945a0SNathan Whitehorn }
23224c8945a0SNathan Whitehorn doupdate();
23234c8945a0SNathan Whitehorn }
23244c8945a0SNathan Whitehorn
23254c8945a0SNathan Whitehorn /*
23264c8945a0SNathan Whitehorn * Create a window, optionally with a shadow.
23274c8945a0SNathan Whitehorn */
23284c8945a0SNathan Whitehorn WINDOW *
dlg_new_window(int height,int width,int y,int x)23294c8945a0SNathan Whitehorn dlg_new_window(int height, int width, int y, int x)
23304c8945a0SNathan Whitehorn {
2331682c9e0fSNathan Whitehorn return dlg_new_modal_window(stdscr, height, width, y, x);
23324c8945a0SNathan Whitehorn }
23334c8945a0SNathan Whitehorn
2334682c9e0fSNathan Whitehorn /*
2335682c9e0fSNathan Whitehorn * "Modal" windows differ from normal ones by having a shadow in a window
2336682c9e0fSNathan Whitehorn * separate from the standard screen.
2337682c9e0fSNathan Whitehorn */
23384c8945a0SNathan Whitehorn WINDOW *
dlg_new_modal_window(WINDOW * parent,int height,int width,int y,int x)23394c8945a0SNathan Whitehorn dlg_new_modal_window(WINDOW *parent, int height, int width, int y, int x)
23404c8945a0SNathan Whitehorn {
23414c8945a0SNathan Whitehorn WINDOW *win;
23424c8945a0SNathan Whitehorn DIALOG_WINDOWS *p = dlg_calloc(DIALOG_WINDOWS, 1);
23434c8945a0SNathan Whitehorn
23444c8945a0SNathan Whitehorn (void) parent;
23452a3e3873SBaptiste Daroussin if (p == 0
23462a3e3873SBaptiste Daroussin || (win = newwin(height, width, y, x)) == 0) {
23474c8945a0SNathan Whitehorn dlg_exiterr("Can't make new window at (%d,%d), size (%d,%d).\n",
23484c8945a0SNathan Whitehorn y, x, height, width);
23494c8945a0SNathan Whitehorn }
23504c8945a0SNathan Whitehorn p->next = dialog_state.all_windows;
23514c8945a0SNathan Whitehorn p->normal = win;
2352a96ef450SBaptiste Daroussin p->getc_timeout = WTIMEOUT_OFF;
23534c8945a0SNathan Whitehorn dialog_state.all_windows = p;
23544c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
23554c8945a0SNathan Whitehorn if (dialog_state.use_shadow) {
2356682c9e0fSNathan Whitehorn p->shadow = parent;
2357682c9e0fSNathan Whitehorn draw_childs_shadow(p);
23584c8945a0SNathan Whitehorn }
23594c8945a0SNathan Whitehorn #endif
23604c8945a0SNathan Whitehorn
23614c8945a0SNathan Whitehorn (void) keypad(win, TRUE);
23624c8945a0SNathan Whitehorn return win;
23634c8945a0SNathan Whitehorn }
23644c8945a0SNathan Whitehorn
23654c8945a0SNathan Whitehorn /*
2366a96ef450SBaptiste Daroussin * dlg_getc() uses the return-value to determine how to handle an ERR return
2367a96ef450SBaptiste Daroussin * from a non-blocking read:
2368a96ef450SBaptiste Daroussin * a) if greater than zero, there was an expired timeout (blocking for a short
2369a96ef450SBaptiste Daroussin * time), or
2370a96ef450SBaptiste Daroussin * b) if zero, it was a non-blocking read, or
2371a96ef450SBaptiste Daroussin * c) if negative, an error occurred on a blocking read.
2372a96ef450SBaptiste Daroussin */
2373a96ef450SBaptiste Daroussin int
dlg_set_timeout(WINDOW * win,bool will_getc)2374a96ef450SBaptiste Daroussin dlg_set_timeout(WINDOW *win, bool will_getc)
2375a96ef450SBaptiste Daroussin {
2376a96ef450SBaptiste Daroussin DIALOG_WINDOWS *p;
2377a96ef450SBaptiste Daroussin int result = 0;
2378a96ef450SBaptiste Daroussin
2379a96ef450SBaptiste Daroussin if ((p = SearchTopWindows(win)) != NULL) {
2380a96ef450SBaptiste Daroussin int interval = (dialog_vars.timeout_secs * 1000);
2381a96ef450SBaptiste Daroussin
2382a96ef450SBaptiste Daroussin if (will_getc || dialog_vars.pause_secs) {
2383a96ef450SBaptiste Daroussin interval = WTIMEOUT_VAL;
2384a96ef450SBaptiste Daroussin } else {
2385a96ef450SBaptiste Daroussin result = interval;
2386a96ef450SBaptiste Daroussin if (interval <= 0) {
2387a96ef450SBaptiste Daroussin interval = WTIMEOUT_OFF;
2388a96ef450SBaptiste Daroussin }
2389a96ef450SBaptiste Daroussin }
2390a96ef450SBaptiste Daroussin wtimeout(win, interval);
2391a96ef450SBaptiste Daroussin p->getc_timeout = interval;
2392a96ef450SBaptiste Daroussin }
2393a96ef450SBaptiste Daroussin return result;
2394a96ef450SBaptiste Daroussin }
2395a96ef450SBaptiste Daroussin
2396a96ef450SBaptiste Daroussin void
dlg_reset_timeout(WINDOW * win)2397a96ef450SBaptiste Daroussin dlg_reset_timeout(WINDOW *win)
2398a96ef450SBaptiste Daroussin {
2399a96ef450SBaptiste Daroussin DIALOG_WINDOWS *p;
2400a96ef450SBaptiste Daroussin
2401a96ef450SBaptiste Daroussin if ((p = SearchTopWindows(win)) != NULL) {
2402a96ef450SBaptiste Daroussin wtimeout(win, p->getc_timeout);
2403a96ef450SBaptiste Daroussin } else {
2404a96ef450SBaptiste Daroussin wtimeout(win, WTIMEOUT_OFF);
2405a96ef450SBaptiste Daroussin }
2406a96ef450SBaptiste Daroussin }
2407a96ef450SBaptiste Daroussin
2408a96ef450SBaptiste Daroussin /*
24094c8945a0SNathan Whitehorn * Move/Resize a window, optionally with a shadow.
24104c8945a0SNathan Whitehorn */
24114c8945a0SNathan Whitehorn #ifdef KEY_RESIZE
24124c8945a0SNathan Whitehorn void
dlg_move_window(WINDOW * win,int height,int width,int y,int x)24134c8945a0SNathan Whitehorn dlg_move_window(WINDOW *win, int height, int width, int y, int x)
24144c8945a0SNathan Whitehorn {
2415a96ef450SBaptiste Daroussin if (win != 0) {
2416682c9e0fSNathan Whitehorn DIALOG_WINDOWS *p;
24174c8945a0SNathan Whitehorn
24184c8945a0SNathan Whitehorn dlg_ctl_size(height, width);
24194c8945a0SNathan Whitehorn
2420a96ef450SBaptiste Daroussin if ((p = SearchTopWindows(win)) != 0) {
24214c8945a0SNathan Whitehorn (void) wresize(win, height, width);
24224c8945a0SNathan Whitehorn (void) mvwin(win, y, x);
24234c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
24244c8945a0SNathan Whitehorn if (p->shadow != 0) {
24254c8945a0SNathan Whitehorn if (dialog_state.use_shadow) {
24264c8945a0SNathan Whitehorn (void) mvwin(p->shadow, y + SHADOW_ROWS, x + SHADOW_COLS);
24274c8945a0SNathan Whitehorn } else {
24284c8945a0SNathan Whitehorn p->shadow = 0;
24294c8945a0SNathan Whitehorn }
24304c8945a0SNathan Whitehorn }
24314c8945a0SNathan Whitehorn #endif
24324c8945a0SNathan Whitehorn (void) refresh();
24334c8945a0SNathan Whitehorn
24344c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
2435682c9e0fSNathan Whitehorn draw_childs_shadow(p);
24364c8945a0SNathan Whitehorn #endif
24374c8945a0SNathan Whitehorn }
24384c8945a0SNathan Whitehorn }
24394c8945a0SNathan Whitehorn }
2440f4f33ea0SBaptiste Daroussin
2441f4f33ea0SBaptiste Daroussin /*
2442f4f33ea0SBaptiste Daroussin * Having just received a KEY_RESIZE, wait a short time to ignore followup
2443f4f33ea0SBaptiste Daroussin * KEY_RESIZE events.
2444f4f33ea0SBaptiste Daroussin */
2445f4f33ea0SBaptiste Daroussin void
dlg_will_resize(WINDOW * win)2446f4f33ea0SBaptiste Daroussin dlg_will_resize(WINDOW *win)
2447f4f33ea0SBaptiste Daroussin {
2448a96ef450SBaptiste Daroussin int n, base;
2449f4f33ea0SBaptiste Daroussin int caught = 0;
2450f4f33ea0SBaptiste Daroussin
2451a96ef450SBaptiste Daroussin dialog_state.had_resize = TRUE;
2452f4f33ea0SBaptiste Daroussin dlg_trace_win(win);
2453a96ef450SBaptiste Daroussin wtimeout(win, WTIMEOUT_VAL * 5);
2454a96ef450SBaptiste Daroussin
2455f4f33ea0SBaptiste Daroussin for (n = base = 0; n < base + 10; ++n) {
2456a96ef450SBaptiste Daroussin int ch;
2457a96ef450SBaptiste Daroussin
2458f4f33ea0SBaptiste Daroussin if ((ch = wgetch(win)) != ERR) {
2459f4f33ea0SBaptiste Daroussin if (ch == KEY_RESIZE) {
2460f4f33ea0SBaptiste Daroussin base = n;
2461f4f33ea0SBaptiste Daroussin ++caught;
2462a96ef450SBaptiste Daroussin } else if (ch != ERR) {
2463f4f33ea0SBaptiste Daroussin ungetch(ch);
2464f4f33ea0SBaptiste Daroussin break;
2465f4f33ea0SBaptiste Daroussin }
2466f4f33ea0SBaptiste Daroussin }
2467f4f33ea0SBaptiste Daroussin }
2468a96ef450SBaptiste Daroussin dlg_reset_timeout(win);
2469a96ef450SBaptiste Daroussin DLG_TRACE(("# caught %d KEY_RESIZE key%s\n",
2470f4f33ea0SBaptiste Daroussin 1 + caught,
2471a96ef450SBaptiste Daroussin caught == 1 ? "" : "s"));
2472f4f33ea0SBaptiste Daroussin }
24734c8945a0SNathan Whitehorn #endif /* KEY_RESIZE */
24744c8945a0SNathan Whitehorn
24754c8945a0SNathan Whitehorn WINDOW *
dlg_der_window(WINDOW * parent,int height,int width,int y,int x)2476a96ef450SBaptiste Daroussin dlg_der_window(WINDOW *parent, int height, int width, int y, int x)
2477a96ef450SBaptiste Daroussin {
2478a96ef450SBaptiste Daroussin WINDOW *win;
2479a96ef450SBaptiste Daroussin
2480a96ef450SBaptiste Daroussin /* existing uses of derwin are (almost) guaranteed to succeed, and the
2481a96ef450SBaptiste Daroussin * caller has to allow for failure.
2482a96ef450SBaptiste Daroussin */
2483a96ef450SBaptiste Daroussin if ((win = derwin(parent, height, width, y, x)) != 0) {
2484a96ef450SBaptiste Daroussin add_subwindow(parent, win);
2485a96ef450SBaptiste Daroussin (void) keypad(win, TRUE);
2486a96ef450SBaptiste Daroussin }
2487a96ef450SBaptiste Daroussin return win;
2488a96ef450SBaptiste Daroussin }
2489a96ef450SBaptiste Daroussin
2490a96ef450SBaptiste Daroussin WINDOW *
dlg_sub_window(WINDOW * parent,int height,int width,int y,int x)24914c8945a0SNathan Whitehorn dlg_sub_window(WINDOW *parent, int height, int width, int y, int x)
24924c8945a0SNathan Whitehorn {
24934c8945a0SNathan Whitehorn WINDOW *win;
24944c8945a0SNathan Whitehorn
24954c8945a0SNathan Whitehorn if ((win = subwin(parent, height, width, y, x)) == 0) {
24964c8945a0SNathan Whitehorn dlg_exiterr("Can't make sub-window at (%d,%d), size (%d,%d).\n",
24974c8945a0SNathan Whitehorn y, x, height, width);
24984c8945a0SNathan Whitehorn }
24994c8945a0SNathan Whitehorn
25002a3e3873SBaptiste Daroussin add_subwindow(parent, win);
25014c8945a0SNathan Whitehorn (void) keypad(win, TRUE);
25024c8945a0SNathan Whitehorn return win;
25034c8945a0SNathan Whitehorn }
25044c8945a0SNathan Whitehorn
25054c8945a0SNathan Whitehorn /* obsolete */
25064c8945a0SNathan Whitehorn int
dlg_default_item(char ** items,int llen)25074c8945a0SNathan Whitehorn dlg_default_item(char **items, int llen)
25084c8945a0SNathan Whitehorn {
25094c8945a0SNathan Whitehorn int result = 0;
25104c8945a0SNathan Whitehorn
25114c8945a0SNathan Whitehorn if (dialog_vars.default_item != 0) {
25124c8945a0SNathan Whitehorn int count = 0;
25134c8945a0SNathan Whitehorn while (*items != 0) {
25144c8945a0SNathan Whitehorn if (!strcmp(dialog_vars.default_item, *items)) {
25154c8945a0SNathan Whitehorn result = count;
25164c8945a0SNathan Whitehorn break;
25174c8945a0SNathan Whitehorn }
25184c8945a0SNathan Whitehorn items += llen;
25194c8945a0SNathan Whitehorn count++;
25204c8945a0SNathan Whitehorn }
25214c8945a0SNathan Whitehorn }
25224c8945a0SNathan Whitehorn return result;
25234c8945a0SNathan Whitehorn }
25244c8945a0SNathan Whitehorn
25254c8945a0SNathan Whitehorn int
dlg_default_listitem(DIALOG_LISTITEM * items)25264c8945a0SNathan Whitehorn dlg_default_listitem(DIALOG_LISTITEM * items)
25274c8945a0SNathan Whitehorn {
25284c8945a0SNathan Whitehorn int result = 0;
25294c8945a0SNathan Whitehorn
25304c8945a0SNathan Whitehorn if (dialog_vars.default_item != 0) {
25314c8945a0SNathan Whitehorn int count = 0;
25324c8945a0SNathan Whitehorn while (items->name != 0) {
25334c8945a0SNathan Whitehorn if (!strcmp(dialog_vars.default_item, items->name)) {
25344c8945a0SNathan Whitehorn result = count;
25354c8945a0SNathan Whitehorn break;
25364c8945a0SNathan Whitehorn }
25374c8945a0SNathan Whitehorn ++items;
25384c8945a0SNathan Whitehorn count++;
25394c8945a0SNathan Whitehorn }
25404c8945a0SNathan Whitehorn }
25414c8945a0SNathan Whitehorn return result;
25424c8945a0SNathan Whitehorn }
25434c8945a0SNathan Whitehorn
25444c8945a0SNathan Whitehorn /*
25454c8945a0SNathan Whitehorn * Draw the string for item_help
25464c8945a0SNathan Whitehorn */
25474c8945a0SNathan Whitehorn void
dlg_item_help(const char * txt)25484c8945a0SNathan Whitehorn dlg_item_help(const char *txt)
25494c8945a0SNathan Whitehorn {
25504c8945a0SNathan Whitehorn if (USE_ITEM_HELP(txt)) {
25514c8945a0SNathan Whitehorn chtype attr = A_NORMAL;
25524c8945a0SNathan Whitehorn
2553f4f33ea0SBaptiste Daroussin dlg_attrset(stdscr, itemhelp_attr);
25544c8945a0SNathan Whitehorn (void) wmove(stdscr, LINES - 1, 0);
25554c8945a0SNathan Whitehorn (void) wclrtoeol(stdscr);
25564c8945a0SNathan Whitehorn (void) addch(' ');
25574c8945a0SNathan Whitehorn dlg_print_text(stdscr, txt, COLS - 1, &attr);
2558a96ef450SBaptiste Daroussin
25594c8945a0SNathan Whitehorn if (itemhelp_attr & A_COLOR) {
2560a96ef450SBaptiste Daroussin int y, x;
25614c8945a0SNathan Whitehorn /* fill the remainder of the line with the window's attributes */
25624c8945a0SNathan Whitehorn getyx(stdscr, y, x);
25632a3e3873SBaptiste Daroussin (void) y;
25644c8945a0SNathan Whitehorn while (x < COLS) {
25654c8945a0SNathan Whitehorn (void) addch(' ');
25664c8945a0SNathan Whitehorn ++x;
25674c8945a0SNathan Whitehorn }
25684c8945a0SNathan Whitehorn }
25694c8945a0SNathan Whitehorn (void) wnoutrefresh(stdscr);
25704c8945a0SNathan Whitehorn }
25714c8945a0SNathan Whitehorn }
25724c8945a0SNathan Whitehorn
25734c8945a0SNathan Whitehorn #ifndef HAVE_STRCASECMP
25744c8945a0SNathan Whitehorn int
dlg_strcmp(const char * a,const char * b)25754c8945a0SNathan Whitehorn dlg_strcmp(const char *a, const char *b)
25764c8945a0SNathan Whitehorn {
25774c8945a0SNathan Whitehorn int ac, bc, cmp;
25784c8945a0SNathan Whitehorn
25794c8945a0SNathan Whitehorn for (;;) {
25804c8945a0SNathan Whitehorn ac = UCH(*a++);
25814c8945a0SNathan Whitehorn bc = UCH(*b++);
25824c8945a0SNathan Whitehorn if (isalpha(ac) && islower(ac))
25834c8945a0SNathan Whitehorn ac = _toupper(ac);
25844c8945a0SNathan Whitehorn if (isalpha(bc) && islower(bc))
25854c8945a0SNathan Whitehorn bc = _toupper(bc);
25864c8945a0SNathan Whitehorn cmp = ac - bc;
25874c8945a0SNathan Whitehorn if (ac == 0 || bc == 0 || cmp != 0)
25884c8945a0SNathan Whitehorn break;
25894c8945a0SNathan Whitehorn }
25904c8945a0SNathan Whitehorn return cmp;
25914c8945a0SNathan Whitehorn }
25924c8945a0SNathan Whitehorn #endif
25934c8945a0SNathan Whitehorn
25944c8945a0SNathan Whitehorn /*
25954c8945a0SNathan Whitehorn * Returns true if 'dst' points to a blank which follows another blank which
25964c8945a0SNathan Whitehorn * is not a leading blank on a line.
25974c8945a0SNathan Whitehorn */
25984c8945a0SNathan Whitehorn static bool
trim_blank(char * base,char * dst)25994c8945a0SNathan Whitehorn trim_blank(char *base, char *dst)
26004c8945a0SNathan Whitehorn {
2601a96ef450SBaptiste Daroussin int count = !!isblank(UCH(*dst));
26024c8945a0SNathan Whitehorn
26034c8945a0SNathan Whitehorn while (dst-- != base) {
26044c8945a0SNathan Whitehorn if (*dst == '\n') {
2605f4f33ea0SBaptiste Daroussin break;
2606f4f33ea0SBaptiste Daroussin } else if (isblank(UCH(*dst))) {
26074c8945a0SNathan Whitehorn count++;
2608f4f33ea0SBaptiste Daroussin } else {
2609f4f33ea0SBaptiste Daroussin break;
26104c8945a0SNathan Whitehorn }
26114c8945a0SNathan Whitehorn }
2612f4f33ea0SBaptiste Daroussin return (count > 1);
26134c8945a0SNathan Whitehorn }
26144c8945a0SNathan Whitehorn
26154c8945a0SNathan Whitehorn /*
26164c8945a0SNathan Whitehorn * Change embedded "\n" substrings to '\n' characters and tabs to single
26174c8945a0SNathan Whitehorn * spaces. If there are no "\n"s, it will strip all extra spaces, for
26184c8945a0SNathan Whitehorn * justification. If it has "\n"'s, it will preserve extra spaces. If cr_wrap
26194c8945a0SNathan Whitehorn * is set, it will preserve '\n's.
26204c8945a0SNathan Whitehorn */
26214c8945a0SNathan Whitehorn void
dlg_trim_string(char * s)26224c8945a0SNathan Whitehorn dlg_trim_string(char *s)
26234c8945a0SNathan Whitehorn {
26244c8945a0SNathan Whitehorn char *base = s;
26254c8945a0SNathan Whitehorn char *p1;
26264c8945a0SNathan Whitehorn char *p = s;
2627682c9e0fSNathan Whitehorn int has_newlines = !dialog_vars.no_nl_expand && (strstr(s, "\\n") != 0);
26284c8945a0SNathan Whitehorn
26294c8945a0SNathan Whitehorn while (*p != '\0') {
26304c8945a0SNathan Whitehorn if (*p == TAB && !dialog_vars.nocollapse)
26314c8945a0SNathan Whitehorn *p = ' ';
26324c8945a0SNathan Whitehorn
26334c8945a0SNathan Whitehorn if (has_newlines) { /* If prompt contains "\n" strings */
26344c8945a0SNathan Whitehorn if (*p == '\\' && *(p + 1) == 'n') {
26354c8945a0SNathan Whitehorn *s++ = '\n';
26364c8945a0SNathan Whitehorn p += 2;
26374c8945a0SNathan Whitehorn p1 = p;
26384c8945a0SNathan Whitehorn /*
26394c8945a0SNathan Whitehorn * Handle end of lines intelligently. If '\n' follows "\n"
26404c8945a0SNathan Whitehorn * then ignore the '\n'. This eliminates the need to escape
26414c8945a0SNathan Whitehorn * the '\n' character (no need to use "\n\").
26424c8945a0SNathan Whitehorn */
2643f4f33ea0SBaptiste Daroussin while (isblank(UCH(*p1)))
26444c8945a0SNathan Whitehorn p1++;
26454c8945a0SNathan Whitehorn if (*p1 == '\n')
26464c8945a0SNathan Whitehorn p = p1 + 1;
26474c8945a0SNathan Whitehorn } else if (*p == '\n') {
26484c8945a0SNathan Whitehorn if (dialog_vars.cr_wrap)
26494c8945a0SNathan Whitehorn *s++ = *p++;
26504c8945a0SNathan Whitehorn else {
26514c8945a0SNathan Whitehorn /* Replace the '\n' with a space if cr_wrap is not set */
2652f4f33ea0SBaptiste Daroussin if (!trim_blank(base, p))
26534c8945a0SNathan Whitehorn *s++ = ' ';
26544c8945a0SNathan Whitehorn p++;
26554c8945a0SNathan Whitehorn }
26564c8945a0SNathan Whitehorn } else /* If *p != '\n' */
26574c8945a0SNathan Whitehorn *s++ = *p++;
26584c8945a0SNathan Whitehorn } else if (dialog_vars.trim_whitespace) {
2659f4f33ea0SBaptiste Daroussin if (isblank(UCH(*p))) {
2660f4f33ea0SBaptiste Daroussin if (!isblank(UCH(*(s - 1)))) {
26614c8945a0SNathan Whitehorn *s++ = ' ';
26624c8945a0SNathan Whitehorn p++;
26634c8945a0SNathan Whitehorn } else
26644c8945a0SNathan Whitehorn p++;
26654c8945a0SNathan Whitehorn } else if (*p == '\n') {
26664c8945a0SNathan Whitehorn if (dialog_vars.cr_wrap)
26674c8945a0SNathan Whitehorn *s++ = *p++;
2668f4f33ea0SBaptiste Daroussin else if (!isblank(UCH(*(s - 1)))) {
26694c8945a0SNathan Whitehorn /* Strip '\n's if cr_wrap is not set. */
26704c8945a0SNathan Whitehorn *s++ = ' ';
26714c8945a0SNathan Whitehorn p++;
26724c8945a0SNathan Whitehorn } else
26734c8945a0SNathan Whitehorn p++;
26744c8945a0SNathan Whitehorn } else
26754c8945a0SNathan Whitehorn *s++ = *p++;
26764c8945a0SNathan Whitehorn } else { /* If there are no "\n" strings */
2677f4f33ea0SBaptiste Daroussin if (isblank(UCH(*p)) && !dialog_vars.nocollapse) {
2678f4f33ea0SBaptiste Daroussin if (!trim_blank(base, p))
26794c8945a0SNathan Whitehorn *s++ = *p;
26804c8945a0SNathan Whitehorn p++;
26814c8945a0SNathan Whitehorn } else
26824c8945a0SNathan Whitehorn *s++ = *p++;
26834c8945a0SNathan Whitehorn }
26844c8945a0SNathan Whitehorn }
26854c8945a0SNathan Whitehorn
26864c8945a0SNathan Whitehorn *s = '\0';
26874c8945a0SNathan Whitehorn }
26884c8945a0SNathan Whitehorn
26894c8945a0SNathan Whitehorn void
dlg_set_focus(WINDOW * parent,WINDOW * win)26904c8945a0SNathan Whitehorn dlg_set_focus(WINDOW *parent, WINDOW *win)
26914c8945a0SNathan Whitehorn {
26924c8945a0SNathan Whitehorn if (win != 0) {
26934c8945a0SNathan Whitehorn (void) wmove(parent,
26944c8945a0SNathan Whitehorn getpary(win) + getcury(win),
26954c8945a0SNathan Whitehorn getparx(win) + getcurx(win));
26964c8945a0SNathan Whitehorn (void) wnoutrefresh(win);
26974c8945a0SNathan Whitehorn (void) doupdate();
26984c8945a0SNathan Whitehorn }
26994c8945a0SNathan Whitehorn }
27004c8945a0SNathan Whitehorn
27014c8945a0SNathan Whitehorn /*
27024c8945a0SNathan Whitehorn * Returns the nominal maximum buffer size.
27034c8945a0SNathan Whitehorn */
27044c8945a0SNathan Whitehorn int
dlg_max_input(int max_len)27054c8945a0SNathan Whitehorn dlg_max_input(int max_len)
27064c8945a0SNathan Whitehorn {
27074c8945a0SNathan Whitehorn if (dialog_vars.max_input != 0 && dialog_vars.max_input < MAX_LEN)
27084c8945a0SNathan Whitehorn max_len = dialog_vars.max_input;
27094c8945a0SNathan Whitehorn
27104c8945a0SNathan Whitehorn return max_len;
27114c8945a0SNathan Whitehorn }
27124c8945a0SNathan Whitehorn
27134c8945a0SNathan Whitehorn /*
27144c8945a0SNathan Whitehorn * Free storage used for the result buffer.
27154c8945a0SNathan Whitehorn */
27164c8945a0SNathan Whitehorn void
dlg_clr_result(void)27174c8945a0SNathan Whitehorn dlg_clr_result(void)
27184c8945a0SNathan Whitehorn {
27194c8945a0SNathan Whitehorn if (dialog_vars.input_length) {
27204c8945a0SNathan Whitehorn dialog_vars.input_length = 0;
27214c8945a0SNathan Whitehorn if (dialog_vars.input_result)
27224c8945a0SNathan Whitehorn free(dialog_vars.input_result);
27234c8945a0SNathan Whitehorn }
27244c8945a0SNathan Whitehorn dialog_vars.input_result = 0;
27254c8945a0SNathan Whitehorn }
27264c8945a0SNathan Whitehorn
27274c8945a0SNathan Whitehorn /*
27284c8945a0SNathan Whitehorn * Setup a fixed-buffer for the result.
27294c8945a0SNathan Whitehorn */
27304c8945a0SNathan Whitehorn char *
dlg_set_result(const char * string)27314c8945a0SNathan Whitehorn dlg_set_result(const char *string)
27324c8945a0SNathan Whitehorn {
27337a1c0d96SNathan Whitehorn unsigned need = string ? (unsigned) strlen(string) + 1 : 0;
27344c8945a0SNathan Whitehorn
27354c8945a0SNathan Whitehorn /* inputstr.c needs a fixed buffer */
27364c8945a0SNathan Whitehorn if (need < MAX_LEN)
27374c8945a0SNathan Whitehorn need = MAX_LEN;
27384c8945a0SNathan Whitehorn
27394c8945a0SNathan Whitehorn /*
27404c8945a0SNathan Whitehorn * If the buffer is not big enough, allocate a new one.
27414c8945a0SNathan Whitehorn */
27424c8945a0SNathan Whitehorn if (dialog_vars.input_length != 0
27434c8945a0SNathan Whitehorn || dialog_vars.input_result == 0
27444c8945a0SNathan Whitehorn || need > MAX_LEN) {
27454c8945a0SNathan Whitehorn
27464c8945a0SNathan Whitehorn dlg_clr_result();
27474c8945a0SNathan Whitehorn
27484c8945a0SNathan Whitehorn dialog_vars.input_length = need;
27494c8945a0SNathan Whitehorn dialog_vars.input_result = dlg_malloc(char, need);
27504c8945a0SNathan Whitehorn assert_ptr(dialog_vars.input_result, "dlg_set_result");
27514c8945a0SNathan Whitehorn }
27524c8945a0SNathan Whitehorn
27534c8945a0SNathan Whitehorn strcpy(dialog_vars.input_result, string ? string : "");
27544c8945a0SNathan Whitehorn
27554c8945a0SNathan Whitehorn return dialog_vars.input_result;
27564c8945a0SNathan Whitehorn }
27574c8945a0SNathan Whitehorn
27584c8945a0SNathan Whitehorn /*
27594c8945a0SNathan Whitehorn * Accumulate results in dynamically allocated buffer.
27604c8945a0SNathan Whitehorn * If input_length is zero, it is a MAX_LEN buffer belonging to the caller.
27614c8945a0SNathan Whitehorn */
27624c8945a0SNathan Whitehorn void
dlg_add_result(const char * string)27634c8945a0SNathan Whitehorn dlg_add_result(const char *string)
27644c8945a0SNathan Whitehorn {
27654c8945a0SNathan Whitehorn unsigned have = (dialog_vars.input_result
27667a1c0d96SNathan Whitehorn ? (unsigned) strlen(dialog_vars.input_result)
27674c8945a0SNathan Whitehorn : 0);
27687a1c0d96SNathan Whitehorn unsigned want = (unsigned) strlen(string) + 1 + have;
27694c8945a0SNathan Whitehorn
27704c8945a0SNathan Whitehorn if ((want >= MAX_LEN)
27714c8945a0SNathan Whitehorn || (dialog_vars.input_length != 0)
27724c8945a0SNathan Whitehorn || (dialog_vars.input_result == 0)) {
27734c8945a0SNathan Whitehorn
27744c8945a0SNathan Whitehorn if (dialog_vars.input_length == 0
27754c8945a0SNathan Whitehorn || dialog_vars.input_result == 0) {
27764c8945a0SNathan Whitehorn
27777a1c0d96SNathan Whitehorn char *save_result = dialog_vars.input_result;
27784c8945a0SNathan Whitehorn
27794c8945a0SNathan Whitehorn dialog_vars.input_length = want * 2;
27804c8945a0SNathan Whitehorn dialog_vars.input_result = dlg_malloc(char, dialog_vars.input_length);
27814c8945a0SNathan Whitehorn assert_ptr(dialog_vars.input_result, "dlg_add_result malloc");
27827a1c0d96SNathan Whitehorn dialog_vars.input_result[0] = '\0';
27837a1c0d96SNathan Whitehorn if (save_result != 0)
27847a1c0d96SNathan Whitehorn strcpy(dialog_vars.input_result, save_result);
27854c8945a0SNathan Whitehorn } else if (want >= dialog_vars.input_length) {
27864c8945a0SNathan Whitehorn dialog_vars.input_length = want * 2;
27874c8945a0SNathan Whitehorn dialog_vars.input_result = dlg_realloc(char,
27884c8945a0SNathan Whitehorn dialog_vars.input_length,
27894c8945a0SNathan Whitehorn dialog_vars.input_result);
27904c8945a0SNathan Whitehorn assert_ptr(dialog_vars.input_result, "dlg_add_result realloc");
27914c8945a0SNathan Whitehorn }
27924c8945a0SNathan Whitehorn }
27934c8945a0SNathan Whitehorn strcat(dialog_vars.input_result, string);
27944c8945a0SNathan Whitehorn }
27954c8945a0SNathan Whitehorn
27964c8945a0SNathan Whitehorn /*
27974c8945a0SNathan Whitehorn * These are characters that (aside from the quote-delimiter) will have to
27984c8945a0SNathan Whitehorn * be escaped in a single- or double-quoted string.
27994c8945a0SNathan Whitehorn */
28004c8945a0SNathan Whitehorn #define FIX_SINGLE "\n\\"
28014c8945a0SNathan Whitehorn #define FIX_DOUBLE FIX_SINGLE "[]{}?*;`~#$^&()|<>"
28024c8945a0SNathan Whitehorn
28034c8945a0SNathan Whitehorn /*
28044c8945a0SNathan Whitehorn * Returns the quote-delimiter.
28054c8945a0SNathan Whitehorn */
28064c8945a0SNathan Whitehorn static const char *
quote_delimiter(void)28074c8945a0SNathan Whitehorn quote_delimiter(void)
28084c8945a0SNathan Whitehorn {
28094c8945a0SNathan Whitehorn return dialog_vars.single_quoted ? "'" : "\"";
28104c8945a0SNathan Whitehorn }
28114c8945a0SNathan Whitehorn
28124c8945a0SNathan Whitehorn /*
28134c8945a0SNathan Whitehorn * Returns true if we should quote the given string.
28144c8945a0SNathan Whitehorn */
28154c8945a0SNathan Whitehorn static bool
must_quote(char * string)28164c8945a0SNathan Whitehorn must_quote(char *string)
28174c8945a0SNathan Whitehorn {
28184c8945a0SNathan Whitehorn bool code = FALSE;
28194c8945a0SNathan Whitehorn
28204c8945a0SNathan Whitehorn if (*string != '\0') {
28217a1c0d96SNathan Whitehorn size_t len = strlen(string);
28224c8945a0SNathan Whitehorn if (strcspn(string, quote_delimiter()) != len)
28234c8945a0SNathan Whitehorn code = TRUE;
28244c8945a0SNathan Whitehorn else if (strcspn(string, "\n\t ") != len)
28254c8945a0SNathan Whitehorn code = TRUE;
28264c8945a0SNathan Whitehorn else
28274c8945a0SNathan Whitehorn code = (strcspn(string, FIX_DOUBLE) != len);
28284c8945a0SNathan Whitehorn } else {
28294c8945a0SNathan Whitehorn code = TRUE;
28304c8945a0SNathan Whitehorn }
28314c8945a0SNathan Whitehorn
28324c8945a0SNathan Whitehorn return code;
28334c8945a0SNathan Whitehorn }
28344c8945a0SNathan Whitehorn
28354c8945a0SNathan Whitehorn /*
28364c8945a0SNathan Whitehorn * Add a quoted string to the result buffer.
28374c8945a0SNathan Whitehorn */
28384c8945a0SNathan Whitehorn void
dlg_add_quoted(char * string)28394c8945a0SNathan Whitehorn dlg_add_quoted(char *string)
28404c8945a0SNathan Whitehorn {
28414c8945a0SNathan Whitehorn char temp[2];
28424c8945a0SNathan Whitehorn const char *my_quote = quote_delimiter();
28434c8945a0SNathan Whitehorn const char *must_fix = (dialog_vars.single_quoted
28444c8945a0SNathan Whitehorn ? FIX_SINGLE
28454c8945a0SNathan Whitehorn : FIX_DOUBLE);
28464c8945a0SNathan Whitehorn
28472a3e3873SBaptiste Daroussin if (must_quote(string)) {
28484c8945a0SNathan Whitehorn temp[1] = '\0';
28494c8945a0SNathan Whitehorn dlg_add_result(my_quote);
28504c8945a0SNathan Whitehorn while (*string != '\0') {
28514c8945a0SNathan Whitehorn temp[0] = *string++;
2852f4f33ea0SBaptiste Daroussin if ((strchr) (my_quote, *temp) || (strchr) (must_fix, *temp))
28534c8945a0SNathan Whitehorn dlg_add_result("\\");
28544c8945a0SNathan Whitehorn dlg_add_result(temp);
28554c8945a0SNathan Whitehorn }
28564c8945a0SNathan Whitehorn dlg_add_result(my_quote);
28574c8945a0SNathan Whitehorn } else {
28584c8945a0SNathan Whitehorn dlg_add_result(string);
28594c8945a0SNathan Whitehorn }
28604c8945a0SNathan Whitehorn }
28614c8945a0SNathan Whitehorn
28624c8945a0SNathan Whitehorn /*
28634c8945a0SNathan Whitehorn * When adding a result, make that depend on whether "--quoted" is used.
28644c8945a0SNathan Whitehorn */
28654c8945a0SNathan Whitehorn void
dlg_add_string(char * string)28664c8945a0SNathan Whitehorn dlg_add_string(char *string)
28674c8945a0SNathan Whitehorn {
28684c8945a0SNathan Whitehorn if (dialog_vars.quoted) {
28694c8945a0SNathan Whitehorn dlg_add_quoted(string);
28704c8945a0SNathan Whitehorn } else {
28714c8945a0SNathan Whitehorn dlg_add_result(string);
28724c8945a0SNathan Whitehorn }
28734c8945a0SNathan Whitehorn }
28744c8945a0SNathan Whitehorn
28754c8945a0SNathan Whitehorn bool
dlg_need_separator(void)28764c8945a0SNathan Whitehorn dlg_need_separator(void)
28774c8945a0SNathan Whitehorn {
28784c8945a0SNathan Whitehorn bool result = FALSE;
28794c8945a0SNathan Whitehorn
28804c8945a0SNathan Whitehorn if (dialog_vars.output_separator) {
28814c8945a0SNathan Whitehorn result = TRUE;
28824c8945a0SNathan Whitehorn } else if (dialog_vars.input_result && *(dialog_vars.input_result)) {
28834c8945a0SNathan Whitehorn result = TRUE;
28844c8945a0SNathan Whitehorn }
28854c8945a0SNathan Whitehorn return result;
28864c8945a0SNathan Whitehorn }
28874c8945a0SNathan Whitehorn
28884c8945a0SNathan Whitehorn void
dlg_add_separator(void)28894c8945a0SNathan Whitehorn dlg_add_separator(void)
28904c8945a0SNathan Whitehorn {
28914c8945a0SNathan Whitehorn const char *separator = (dialog_vars.separate_output) ? "\n" : " ";
28924c8945a0SNathan Whitehorn
28934c8945a0SNathan Whitehorn if (dialog_vars.output_separator)
28944c8945a0SNathan Whitehorn separator = dialog_vars.output_separator;
28954c8945a0SNathan Whitehorn
28964c8945a0SNathan Whitehorn dlg_add_result(separator);
28974c8945a0SNathan Whitehorn }
28984c8945a0SNathan Whitehorn
2899febdb468SDevin Teske #define HELP_PREFIX "HELP "
2900febdb468SDevin Teske
2901febdb468SDevin Teske void
dlg_add_help_listitem(int * result,char ** tag,DIALOG_LISTITEM * item)2902febdb468SDevin Teske dlg_add_help_listitem(int *result, char **tag, DIALOG_LISTITEM * item)
2903febdb468SDevin Teske {
2904febdb468SDevin Teske dlg_add_result(HELP_PREFIX);
2905febdb468SDevin Teske if (USE_ITEM_HELP(item->help)) {
2906febdb468SDevin Teske *tag = dialog_vars.help_tags ? item->name : item->help;
2907febdb468SDevin Teske *result = DLG_EXIT_ITEM_HELP;
2908febdb468SDevin Teske } else {
2909febdb468SDevin Teske *tag = item->name;
2910febdb468SDevin Teske }
2911febdb468SDevin Teske }
2912febdb468SDevin Teske
2913febdb468SDevin Teske void
dlg_add_help_formitem(int * result,char ** tag,DIALOG_FORMITEM * item)2914febdb468SDevin Teske dlg_add_help_formitem(int *result, char **tag, DIALOG_FORMITEM * item)
2915febdb468SDevin Teske {
2916febdb468SDevin Teske dlg_add_result(HELP_PREFIX);
2917febdb468SDevin Teske if (USE_ITEM_HELP(item->help)) {
2918febdb468SDevin Teske *tag = dialog_vars.help_tags ? item->name : item->help;
2919febdb468SDevin Teske *result = DLG_EXIT_ITEM_HELP;
2920febdb468SDevin Teske } else {
2921febdb468SDevin Teske *tag = item->name;
2922febdb468SDevin Teske }
2923febdb468SDevin Teske }
2924febdb468SDevin Teske
29254c8945a0SNathan Whitehorn /*
29264c8945a0SNathan Whitehorn * Some widgets support only one value of a given variable - save/restore the
29274c8945a0SNathan Whitehorn * global dialog_vars so we can override it consistently.
29284c8945a0SNathan Whitehorn */
29294c8945a0SNathan Whitehorn void
dlg_save_vars(DIALOG_VARS * vars)29304c8945a0SNathan Whitehorn dlg_save_vars(DIALOG_VARS * vars)
29314c8945a0SNathan Whitehorn {
29324c8945a0SNathan Whitehorn *vars = dialog_vars;
29334c8945a0SNathan Whitehorn }
29344c8945a0SNathan Whitehorn
29357a1c0d96SNathan Whitehorn /*
29367a1c0d96SNathan Whitehorn * Most of the data in DIALOG_VARS is normally set by command-line options.
29377a1c0d96SNathan Whitehorn * The input_result member is an exception; it is normally set by the dialog
29387a1c0d96SNathan Whitehorn * library to return result values.
29397a1c0d96SNathan Whitehorn */
29404c8945a0SNathan Whitehorn void
dlg_restore_vars(DIALOG_VARS * vars)29414c8945a0SNathan Whitehorn dlg_restore_vars(DIALOG_VARS * vars)
29424c8945a0SNathan Whitehorn {
29437a1c0d96SNathan Whitehorn char *save_result = dialog_vars.input_result;
29447a1c0d96SNathan Whitehorn unsigned save_length = dialog_vars.input_length;
29457a1c0d96SNathan Whitehorn
29464c8945a0SNathan Whitehorn dialog_vars = *vars;
29477a1c0d96SNathan Whitehorn dialog_vars.input_result = save_result;
29487a1c0d96SNathan Whitehorn dialog_vars.input_length = save_length;
29494c8945a0SNathan Whitehorn }
29504c8945a0SNathan Whitehorn
29514c8945a0SNathan Whitehorn /*
29524c8945a0SNathan Whitehorn * Called each time a widget is invoked which may do output, increment a count.
29534c8945a0SNathan Whitehorn */
29544c8945a0SNathan Whitehorn void
dlg_does_output(void)29554c8945a0SNathan Whitehorn dlg_does_output(void)
29564c8945a0SNathan Whitehorn {
29574c8945a0SNathan Whitehorn dialog_state.output_count += 1;
29584c8945a0SNathan Whitehorn }
29594c8945a0SNathan Whitehorn
29604c8945a0SNathan Whitehorn /*
29614c8945a0SNathan Whitehorn * Compatibility for different versions of curses.
29624c8945a0SNathan Whitehorn */
29634c8945a0SNathan Whitehorn #if !(defined(HAVE_GETBEGX) && defined(HAVE_GETBEGY))
29644c8945a0SNathan Whitehorn int
dlg_getbegx(WINDOW * win)29652a3e3873SBaptiste Daroussin dlg_getbegx(WINDOW *win)
29664c8945a0SNathan Whitehorn {
29674c8945a0SNathan Whitehorn int y, x;
29684c8945a0SNathan Whitehorn getbegyx(win, y, x);
29694c8945a0SNathan Whitehorn return x;
29704c8945a0SNathan Whitehorn }
29714c8945a0SNathan Whitehorn int
dlg_getbegy(WINDOW * win)29722a3e3873SBaptiste Daroussin dlg_getbegy(WINDOW *win)
29734c8945a0SNathan Whitehorn {
29744c8945a0SNathan Whitehorn int y, x;
29754c8945a0SNathan Whitehorn getbegyx(win, y, x);
29764c8945a0SNathan Whitehorn return y;
29774c8945a0SNathan Whitehorn }
29784c8945a0SNathan Whitehorn #endif
29794c8945a0SNathan Whitehorn
29804c8945a0SNathan Whitehorn #if !(defined(HAVE_GETCURX) && defined(HAVE_GETCURY))
29814c8945a0SNathan Whitehorn int
dlg_getcurx(WINDOW * win)29822a3e3873SBaptiste Daroussin dlg_getcurx(WINDOW *win)
29834c8945a0SNathan Whitehorn {
29844c8945a0SNathan Whitehorn int y, x;
29854c8945a0SNathan Whitehorn getyx(win, y, x);
29864c8945a0SNathan Whitehorn return x;
29874c8945a0SNathan Whitehorn }
29884c8945a0SNathan Whitehorn int
dlg_getcury(WINDOW * win)29892a3e3873SBaptiste Daroussin dlg_getcury(WINDOW *win)
29904c8945a0SNathan Whitehorn {
29914c8945a0SNathan Whitehorn int y, x;
29924c8945a0SNathan Whitehorn getyx(win, y, x);
29934c8945a0SNathan Whitehorn return y;
29944c8945a0SNathan Whitehorn }
29954c8945a0SNathan Whitehorn #endif
29964c8945a0SNathan Whitehorn
29974c8945a0SNathan Whitehorn #if !(defined(HAVE_GETMAXX) && defined(HAVE_GETMAXY))
29984c8945a0SNathan Whitehorn int
dlg_getmaxx(WINDOW * win)29992a3e3873SBaptiste Daroussin dlg_getmaxx(WINDOW *win)
30004c8945a0SNathan Whitehorn {
30014c8945a0SNathan Whitehorn int y, x;
30024c8945a0SNathan Whitehorn getmaxyx(win, y, x);
30034c8945a0SNathan Whitehorn return x;
30044c8945a0SNathan Whitehorn }
30054c8945a0SNathan Whitehorn int
dlg_getmaxy(WINDOW * win)30062a3e3873SBaptiste Daroussin dlg_getmaxy(WINDOW *win)
30074c8945a0SNathan Whitehorn {
30084c8945a0SNathan Whitehorn int y, x;
30094c8945a0SNathan Whitehorn getmaxyx(win, y, x);
30104c8945a0SNathan Whitehorn return y;
30114c8945a0SNathan Whitehorn }
30124c8945a0SNathan Whitehorn #endif
30134c8945a0SNathan Whitehorn
30144c8945a0SNathan Whitehorn #if !(defined(HAVE_GETPARX) && defined(HAVE_GETPARY))
30154c8945a0SNathan Whitehorn int
dlg_getparx(WINDOW * win)30162a3e3873SBaptiste Daroussin dlg_getparx(WINDOW *win)
30174c8945a0SNathan Whitehorn {
30184c8945a0SNathan Whitehorn int y, x;
30194c8945a0SNathan Whitehorn getparyx(win, y, x);
30204c8945a0SNathan Whitehorn return x;
30214c8945a0SNathan Whitehorn }
30224c8945a0SNathan Whitehorn int
dlg_getpary(WINDOW * win)30232a3e3873SBaptiste Daroussin dlg_getpary(WINDOW *win)
30244c8945a0SNathan Whitehorn {
30254c8945a0SNathan Whitehorn int y, x;
30264c8945a0SNathan Whitehorn getparyx(win, y, x);
30274c8945a0SNathan Whitehorn return y;
30284c8945a0SNathan Whitehorn }
30294c8945a0SNathan Whitehorn #endif
30302a3e3873SBaptiste Daroussin
30312a3e3873SBaptiste Daroussin #ifdef NEED_WGETPARENT
30322a3e3873SBaptiste Daroussin WINDOW *
dlg_wgetparent(WINDOW * win)30332a3e3873SBaptiste Daroussin dlg_wgetparent(WINDOW *win)
30342a3e3873SBaptiste Daroussin {
30352a3e3873SBaptiste Daroussin #undef wgetparent
30362a3e3873SBaptiste Daroussin WINDOW *result = 0;
30372a3e3873SBaptiste Daroussin DIALOG_WINDOWS *p;
30382a3e3873SBaptiste Daroussin
30392a3e3873SBaptiste Daroussin for (p = dialog_state.all_subwindows; p != 0; p = p->next) {
30402a3e3873SBaptiste Daroussin if (p->shadow == win) {
30412a3e3873SBaptiste Daroussin result = p->normal;
30422a3e3873SBaptiste Daroussin break;
30432a3e3873SBaptiste Daroussin }
30442a3e3873SBaptiste Daroussin }
30452a3e3873SBaptiste Daroussin return result;
30462a3e3873SBaptiste Daroussin }
30472a3e3873SBaptiste Daroussin #endif
3048