mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-07 13:14:51 +00:00
febdb46880
Approved by: re (marius)
2709 lines
63 KiB
C
2709 lines
63 KiB
C
/*
|
|
* $Id: util.c,v 1.258 2013/09/22 00:41:40 tom Exp $
|
|
*
|
|
* util.c -- miscellaneous utilities for dialog
|
|
*
|
|
* Copyright 2000-2012,2013 Thomas E. Dickey
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License, version 2.1
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this program; if not, write to
|
|
* Free Software Foundation, Inc.
|
|
* 51 Franklin St., Fifth Floor
|
|
* Boston, MA 02110, USA.
|
|
*
|
|
* An earlier version of this program lists as authors
|
|
* Savio Lam (lam836@cs.cuhk.hk)
|
|
*/
|
|
|
|
#include <dialog.h>
|
|
#include <dlg_keys.h>
|
|
|
|
#ifdef HAVE_SETLOCALE
|
|
#include <locale.h>
|
|
#endif
|
|
|
|
#ifdef NEED_WCHAR_H
|
|
#include <wchar.h>
|
|
#endif
|
|
|
|
#ifdef NCURSES_VERSION
|
|
#if defined(HAVE_NCURSESW_TERM_H)
|
|
#include <ncursesw/term.h>
|
|
#elif defined(HAVE_NCURSES_TERM_H)
|
|
#include <ncurses/term.h>
|
|
#else
|
|
#include <term.h>
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(HAVE_WCHGAT)
|
|
# if defined(NCURSES_VERSION_PATCH)
|
|
# if NCURSES_VERSION_PATCH >= 20060715
|
|
# define USE_WCHGAT 1
|
|
# else
|
|
# define USE_WCHGAT 0
|
|
# endif
|
|
# else
|
|
# define USE_WCHGAT 1
|
|
# endif
|
|
#else
|
|
# define USE_WCHGAT 0
|
|
#endif
|
|
|
|
/* globals */
|
|
DIALOG_STATE dialog_state;
|
|
DIALOG_VARS dialog_vars;
|
|
|
|
#if !(defined(HAVE_WGETPARENT) && defined(HAVE_WINDOW__PARENT))
|
|
#define NEED_WGETPARENT 1
|
|
#else
|
|
#undef NEED_WGETPARENT
|
|
#endif
|
|
|
|
#define concat(a,b) a##b
|
|
|
|
#ifdef HAVE_RC_FILE
|
|
#define RC_DATA(name,comment) , #name "_color", comment " color"
|
|
#else
|
|
#define RC_DATA(name,comment) /*nothing */
|
|
#endif
|
|
|
|
#ifdef HAVE_COLOR
|
|
#include <dlg_colors.h>
|
|
#define COLOR_DATA(upr) , \
|
|
concat(DLGC_FG_,upr), \
|
|
concat(DLGC_BG_,upr), \
|
|
concat(DLGC_HL_,upr)
|
|
#else
|
|
#define COLOR_DATA(upr) /*nothing */
|
|
#endif
|
|
|
|
#define DATA(atr,upr,lwr,cmt) { atr COLOR_DATA(upr) RC_DATA(lwr,cmt) }
|
|
|
|
#define UseShadow(dw) ((dw) != 0 && (dw)->normal != 0 && (dw)->shadow != 0)
|
|
|
|
/*
|
|
* Table of color and attribute values, default is for mono display.
|
|
* The order matches the DIALOG_ATR() values.
|
|
*/
|
|
/* *INDENT-OFF* */
|
|
DIALOG_COLORS dlg_color_table[] =
|
|
{
|
|
DATA(A_NORMAL, SCREEN, screen, "Screen"),
|
|
DATA(A_NORMAL, SHADOW, shadow, "Shadow"),
|
|
DATA(A_REVERSE, DIALOG, dialog, "Dialog box"),
|
|
DATA(A_REVERSE, TITLE, title, "Dialog box title"),
|
|
DATA(A_REVERSE, BORDER, border, "Dialog box border"),
|
|
DATA(A_BOLD, BUTTON_ACTIVE, button_active, "Active button"),
|
|
DATA(A_DIM, BUTTON_INACTIVE, button_inactive, "Inactive button"),
|
|
DATA(A_UNDERLINE, BUTTON_KEY_ACTIVE, button_key_active, "Active button key"),
|
|
DATA(A_UNDERLINE, BUTTON_KEY_INACTIVE, button_key_inactive, "Inactive button key"),
|
|
DATA(A_NORMAL, BUTTON_LABEL_ACTIVE, button_label_active, "Active button label"),
|
|
DATA(A_NORMAL, BUTTON_LABEL_INACTIVE, button_label_inactive, "Inactive button label"),
|
|
DATA(A_REVERSE, INPUTBOX, inputbox, "Input box"),
|
|
DATA(A_REVERSE, INPUTBOX_BORDER, inputbox_border, "Input box border"),
|
|
DATA(A_REVERSE, SEARCHBOX, searchbox, "Search box"),
|
|
DATA(A_REVERSE, SEARCHBOX_TITLE, searchbox_title, "Search box title"),
|
|
DATA(A_REVERSE, SEARCHBOX_BORDER, searchbox_border, "Search box border"),
|
|
DATA(A_REVERSE, POSITION_INDICATOR, position_indicator, "File position indicator"),
|
|
DATA(A_REVERSE, MENUBOX, menubox, "Menu box"),
|
|
DATA(A_REVERSE, MENUBOX_BORDER, menubox_border, "Menu box border"),
|
|
DATA(A_REVERSE, ITEM, item, "Item"),
|
|
DATA(A_NORMAL, ITEM_SELECTED, item_selected, "Selected item"),
|
|
DATA(A_REVERSE, TAG, tag, "Tag"),
|
|
DATA(A_REVERSE, TAG_SELECTED, tag_selected, "Selected tag"),
|
|
DATA(A_NORMAL, TAG_KEY, tag_key, "Tag key"),
|
|
DATA(A_BOLD, TAG_KEY_SELECTED, tag_key_selected, "Selected tag key"),
|
|
DATA(A_REVERSE, CHECK, check, "Check box"),
|
|
DATA(A_REVERSE, CHECK_SELECTED, check_selected, "Selected check box"),
|
|
DATA(A_REVERSE, UARROW, uarrow, "Up arrow"),
|
|
DATA(A_REVERSE, DARROW, darrow, "Down arrow"),
|
|
DATA(A_NORMAL, ITEMHELP, itemhelp, "Item help-text"),
|
|
DATA(A_BOLD, FORM_ACTIVE_TEXT, form_active_text, "Active form text"),
|
|
DATA(A_REVERSE, FORM_TEXT, form_text, "Form text"),
|
|
DATA(A_NORMAL, FORM_ITEM_READONLY, form_item_readonly, "Readonly form item"),
|
|
DATA(A_REVERSE, GAUGE, gauge, "Dialog box gauge"),
|
|
DATA(A_REVERSE, BORDER2, border2, "Dialog box border2"),
|
|
DATA(A_REVERSE, INPUTBOX_BORDER2, inputbox_border2, "Input box border2"),
|
|
DATA(A_REVERSE, SEARCHBOX_BORDER2, searchbox_border2, "Search box border2"),
|
|
DATA(A_REVERSE, MENUBOX_BORDER2, menubox_border2, "Menu box border2")
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
/*
|
|
* Maintain a list of subwindows so that we can delete them to cleanup.
|
|
* More important, this provides a fallback when wgetparent() is not available.
|
|
*/
|
|
static void
|
|
add_subwindow(WINDOW *parent, WINDOW *child)
|
|
{
|
|
DIALOG_WINDOWS *p = dlg_calloc(DIALOG_WINDOWS, 1);
|
|
|
|
if (p != 0) {
|
|
p->normal = parent;
|
|
p->shadow = child;
|
|
p->next = dialog_state.all_subwindows;
|
|
dialog_state.all_subwindows = p;
|
|
}
|
|
}
|
|
|
|
static void
|
|
del_subwindows(WINDOW *parent)
|
|
{
|
|
DIALOG_WINDOWS *p = dialog_state.all_subwindows;
|
|
DIALOG_WINDOWS *q = 0;
|
|
DIALOG_WINDOWS *r;
|
|
|
|
while (p != 0) {
|
|
if (p->normal == parent) {
|
|
delwin(p->shadow);
|
|
r = p->next;
|
|
if (q == 0) {
|
|
dialog_state.all_subwindows = r;
|
|
} else {
|
|
q->next = r;
|
|
}
|
|
free(p);
|
|
p = r;
|
|
} else {
|
|
q = p;
|
|
p = p->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Display background title if it exists ...
|
|
*/
|
|
void
|
|
dlg_put_backtitle(void)
|
|
{
|
|
int i;
|
|
|
|
if (dialog_vars.backtitle != NULL) {
|
|
chtype attr = A_NORMAL;
|
|
int backwidth = dlg_count_columns(dialog_vars.backtitle);
|
|
|
|
(void) wattrset(stdscr, screen_attr);
|
|
(void) wmove(stdscr, 0, 1);
|
|
dlg_print_text(stdscr, dialog_vars.backtitle, COLS - 2, &attr);
|
|
for (i = 0; i < COLS - backwidth; i++)
|
|
(void) waddch(stdscr, ' ');
|
|
(void) wmove(stdscr, 1, 1);
|
|
for (i = 0; i < COLS - 2; i++)
|
|
(void) waddch(stdscr, dlg_boxchar(ACS_HLINE));
|
|
}
|
|
|
|
(void) wnoutrefresh(stdscr);
|
|
}
|
|
|
|
/*
|
|
* Set window to attribute 'attr'. There are more efficient ways to do this,
|
|
* but will not work on older/buggy ncurses versions.
|
|
*/
|
|
void
|
|
dlg_attr_clear(WINDOW *win, int height, int width, chtype attr)
|
|
{
|
|
int i, j;
|
|
|
|
(void) wattrset(win, attr);
|
|
for (i = 0; i < height; i++) {
|
|
(void) wmove(win, i, 0);
|
|
for (j = 0; j < width; j++)
|
|
(void) waddch(win, ' ');
|
|
}
|
|
(void) touchwin(win);
|
|
}
|
|
|
|
void
|
|
dlg_clear(void)
|
|
{
|
|
dlg_attr_clear(stdscr, LINES, COLS, screen_attr);
|
|
}
|
|
|
|
#define isprivate(s) ((s) != 0 && strstr(s, "\033[?") != 0)
|
|
|
|
#define TTY_DEVICE "/dev/tty"
|
|
|
|
/*
|
|
* If $DIALOG_TTY exists, allow the program to try to open the terminal
|
|
* directly when stdout is redirected. By default we require the "--stdout"
|
|
* option to be given, but some scripts were written making use of the
|
|
* behavior of dialog which tried opening the terminal anyway.
|
|
*/
|
|
static char *
|
|
dialog_tty(void)
|
|
{
|
|
char *result = getenv("DIALOG_TTY");
|
|
if (result != 0 && atoi(result) == 0)
|
|
result = 0;
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Open the terminal directly. If one of stdin, stdout or stderr really points
|
|
* to a tty, use it. Otherwise give up and open /dev/tty.
|
|
*/
|
|
static int
|
|
open_terminal(char **result, int mode)
|
|
{
|
|
const char *device = TTY_DEVICE;
|
|
if (!isatty(fileno(stderr))
|
|
|| (device = ttyname(fileno(stderr))) == 0) {
|
|
if (!isatty(fileno(stdout))
|
|
|| (device = ttyname(fileno(stdout))) == 0) {
|
|
if (!isatty(fileno(stdin))
|
|
|| (device = ttyname(fileno(stdin))) == 0) {
|
|
device = TTY_DEVICE;
|
|
}
|
|
}
|
|
}
|
|
*result = dlg_strclone(device);
|
|
return open(device, mode);
|
|
}
|
|
|
|
#ifdef NCURSES_VERSION
|
|
static int
|
|
my_putc(int ch)
|
|
{
|
|
char buffer[2];
|
|
int fd = fileno(dialog_state.screen_output);
|
|
|
|
buffer[0] = (char) ch;
|
|
return (int) write(fd, buffer, (size_t) 1);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Do some initialization for dialog.
|
|
*
|
|
* 'input' is the real tty input of dialog. Usually it is stdin, but if
|
|
* --input-fd option is used, it may be anything.
|
|
*
|
|
* 'output' is where dialog will send its result. Usually it is stderr, but
|
|
* if --stdout or --output-fd is used, it may be anything. We are concerned
|
|
* mainly with the case where it happens to be the same as stdout.
|
|
*/
|
|
void
|
|
init_dialog(FILE *input, FILE *output)
|
|
{
|
|
int fd1, fd2;
|
|
char *device = 0;
|
|
|
|
setlocale(LC_ALL, "");
|
|
|
|
dialog_state.output = output;
|
|
dialog_state.tab_len = TAB_LEN;
|
|
dialog_state.aspect_ratio = DEFAULT_ASPECT_RATIO;
|
|
#ifdef HAVE_COLOR
|
|
dialog_state.use_colors = USE_COLORS; /* use colors by default? */
|
|
dialog_state.use_shadow = USE_SHADOW; /* shadow dialog boxes by default? */
|
|
#endif
|
|
|
|
#ifdef HAVE_RC_FILE
|
|
if (dlg_parse_rc() == -1) /* Read the configuration file */
|
|
dlg_exiterr("init_dialog: dlg_parse_rc");
|
|
#endif
|
|
|
|
/*
|
|
* Some widgets (such as gauge) may read from the standard input. Pipes
|
|
* only connect stdout/stdin, so there is not much choice. But reading a
|
|
* pipe would get in the way of curses' normal reading stdin for getch.
|
|
*
|
|
* As in the --stdout (see below), reopening the terminal does not always
|
|
* work properly. dialog provides a --pipe-fd option for this purpose. We
|
|
* test that case first (differing fileno's for input/stdin). If the
|
|
* fileno's are equal, but we're not reading from a tty, see if we can open
|
|
* /dev/tty.
|
|
*/
|
|
dialog_state.pipe_input = stdin;
|
|
if (fileno(input) != fileno(stdin)) {
|
|
if ((fd1 = dup(fileno(input))) >= 0
|
|
&& (fd2 = dup(fileno(stdin))) >= 0) {
|
|
(void) dup2(fileno(input), fileno(stdin));
|
|
dialog_state.pipe_input = fdopen(fd2, "r");
|
|
if (fileno(stdin) != 0) /* some functions may read fd #0 */
|
|
(void) dup2(fileno(stdin), 0);
|
|
} else {
|
|
dlg_exiterr("cannot open tty-input");
|
|
}
|
|
close(fd1);
|
|
} else if (!isatty(fileno(stdin))) {
|
|
if ((fd1 = open_terminal(&device, O_RDONLY)) >= 0) {
|
|
if ((fd2 = dup(fileno(stdin))) >= 0) {
|
|
dialog_state.pipe_input = fdopen(fd2, "r");
|
|
if (freopen(device, "r", stdin) == 0)
|
|
dlg_exiterr("cannot open tty-input");
|
|
if (fileno(stdin) != 0) /* some functions may read fd #0 */
|
|
(void) dup2(fileno(stdin), 0);
|
|
}
|
|
close(fd1);
|
|
}
|
|
free(device);
|
|
}
|
|
|
|
/*
|
|
* If stdout is not a tty and dialog is called with the --stdout option, we
|
|
* have to provide for a way to write to the screen.
|
|
*
|
|
* The curses library normally writes its output to stdout, leaving stderr
|
|
* free for scripting. Scripts are simpler when stdout is redirected. The
|
|
* newterm function is useful; it allows us to specify where the output
|
|
* goes. Reopening the terminal is not portable since several
|
|
* configurations do not allow this to work properly:
|
|
*
|
|
* a) some getty implementations (and possibly broken tty drivers, e.g., on
|
|
* HPUX 10 and 11) cause stdin to act as if it is still in cooked mode
|
|
* even though results from ioctl's state that it is successfully
|
|
* altered to raw mode. Broken is the proper term.
|
|
*
|
|
* b) the user may not have permissions on the device, e.g., if one su's
|
|
* from the login user to another non-privileged user.
|
|
*/
|
|
if (!isatty(fileno(stdout))
|
|
&& (fileno(stdout) == fileno(output) || dialog_tty())) {
|
|
if ((fd1 = open_terminal(&device, O_WRONLY)) >= 0
|
|
&& (dialog_state.screen_output = fdopen(fd1, "w")) != 0) {
|
|
if (newterm(NULL, dialog_state.screen_output, stdin) == 0) {
|
|
dlg_exiterr("cannot initialize curses");
|
|
}
|
|
free(device);
|
|
} else {
|
|
dlg_exiterr("cannot open tty-output");
|
|
}
|
|
} else {
|
|
dialog_state.screen_output = stdout;
|
|
(void) initscr();
|
|
}
|
|
#ifdef NCURSES_VERSION
|
|
/*
|
|
* Cancel xterm's alternate-screen mode.
|
|
*/
|
|
if (!dialog_vars.keep_tite
|
|
&& (fileno(dialog_state.screen_output) != fileno(stdout)
|
|
|| isatty(fileno(dialog_state.screen_output)))
|
|
&& key_mouse != 0 /* xterm and kindred */
|
|
&& isprivate(enter_ca_mode)
|
|
&& isprivate(exit_ca_mode)) {
|
|
/*
|
|
* initscr() or newterm() already wrote enter_ca_mode as a side
|
|
* effect of initializing the screen. It would be nice to not even
|
|
* do that, but we do not really have access to the correct copy of
|
|
* the terminfo description until those functions have been invoked.
|
|
*/
|
|
(void) refresh();
|
|
(void) tputs(exit_ca_mode, 0, my_putc);
|
|
(void) tputs(clear_screen, 0, my_putc);
|
|
/*
|
|
* Prevent ncurses from switching "back" to the normal screen when
|
|
* exiting from dialog. That would move the cursor to the original
|
|
* location saved in xterm. Normally curses sets the cursor position
|
|
* to the first line after the display, but the alternate screen
|
|
* switching is done after that point.
|
|
*
|
|
* Cancelling the strings altogether also works around the buggy
|
|
* implementation of alternate-screen in rxvt, etc., which clear
|
|
* more of the display than they should.
|
|
*/
|
|
enter_ca_mode = 0;
|
|
exit_ca_mode = 0;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_FLUSHINP
|
|
(void) flushinp();
|
|
#endif
|
|
(void) keypad(stdscr, TRUE);
|
|
(void) cbreak();
|
|
(void) noecho();
|
|
|
|
if (!dialog_state.no_mouse) {
|
|
mouse_open();
|
|
}
|
|
|
|
dialog_state.screen_initialized = TRUE;
|
|
|
|
#ifdef HAVE_COLOR
|
|
if (dialog_state.use_colors || dialog_state.use_shadow)
|
|
dlg_color_setup(); /* Set up colors */
|
|
#endif
|
|
|
|
/* Set screen to screen attribute */
|
|
dlg_clear();
|
|
}
|
|
|
|
#ifdef HAVE_COLOR
|
|
static int defined_colors = 1; /* pair-0 is reserved */
|
|
/*
|
|
* Setup for color display
|
|
*/
|
|
void
|
|
dlg_color_setup(void)
|
|
{
|
|
unsigned i;
|
|
|
|
if (has_colors()) { /* Terminal supports color? */
|
|
(void) start_color();
|
|
|
|
#if defined(HAVE_USE_DEFAULT_COLORS)
|
|
use_default_colors();
|
|
#endif
|
|
|
|
#if defined(__NetBSD__) && defined(_CURSES_)
|
|
#define C_ATTR(x,y) (((x) != 0 ? A_BOLD : 0) | COLOR_PAIR((y)))
|
|
/* work around bug in NetBSD curses */
|
|
for (i = 0; i < sizeof(dlg_color_table) /
|
|
sizeof(dlg_color_table[0]); i++) {
|
|
|
|
/* Initialize color pairs */
|
|
(void) init_pair(i + 1,
|
|
dlg_color_table[i].fg,
|
|
dlg_color_table[i].bg);
|
|
|
|
/* Setup color attributes */
|
|
dlg_color_table[i].atr = C_ATTR(dlg_color_table[i].hilite, i + 1);
|
|
}
|
|
defined_colors = i + 1;
|
|
#else
|
|
for (i = 0; i < sizeof(dlg_color_table) /
|
|
sizeof(dlg_color_table[0]); i++) {
|
|
|
|
/* Initialize color pairs */
|
|
chtype color = dlg_color_pair(dlg_color_table[i].fg,
|
|
dlg_color_table[i].bg);
|
|
|
|
/* Setup color attributes */
|
|
dlg_color_table[i].atr = ((dlg_color_table[i].hilite
|
|
? A_BOLD
|
|
: 0)
|
|
| color);
|
|
}
|
|
#endif
|
|
} else {
|
|
dialog_state.use_colors = FALSE;
|
|
dialog_state.use_shadow = FALSE;
|
|
}
|
|
}
|
|
|
|
int
|
|
dlg_color_count(void)
|
|
{
|
|
return sizeof(dlg_color_table) / sizeof(dlg_color_table[0]);
|
|
}
|
|
|
|
/*
|
|
* Wrapper for getattrs(), or the more cumbersome X/Open wattr_get().
|
|
*/
|
|
chtype
|
|
dlg_get_attrs(WINDOW *win)
|
|
{
|
|
chtype result;
|
|
#ifdef HAVE_GETATTRS
|
|
result = (chtype) getattrs(win);
|
|
#else
|
|
attr_t my_result;
|
|
short my_pair;
|
|
wattr_get(win, &my_result, &my_pair, NULL);
|
|
result = my_result;
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Reuse color pairs (they are limited), returning a COLOR_PAIR() value if we
|
|
* have (or can) define a pair with the given color as foreground on the
|
|
* window's defined background.
|
|
*/
|
|
chtype
|
|
dlg_color_pair(int foreground, int background)
|
|
{
|
|
chtype result = 0;
|
|
int pair;
|
|
short fg, bg;
|
|
bool found = FALSE;
|
|
|
|
for (pair = 1; pair < defined_colors; ++pair) {
|
|
if (pair_content((short) pair, &fg, &bg) != ERR
|
|
&& fg == foreground
|
|
&& bg == background) {
|
|
result = (chtype) COLOR_PAIR(pair);
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
if (!found && (defined_colors + 1) < COLOR_PAIRS) {
|
|
pair = defined_colors++;
|
|
(void) init_pair((short) pair, (short) foreground, (short) background);
|
|
result = (chtype) COLOR_PAIR(pair);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Reuse color pairs (they are limited), returning a COLOR_PAIR() value if we
|
|
* have (or can) define a pair with the given color as foreground on the
|
|
* window's defined background.
|
|
*/
|
|
static chtype
|
|
define_color(WINDOW *win, int foreground)
|
|
{
|
|
chtype attrs = dlg_get_attrs(win);
|
|
int pair;
|
|
short fg, bg, background;
|
|
|
|
if ((pair = PAIR_NUMBER(attrs)) != 0
|
|
&& pair_content((short) pair, &fg, &bg) != ERR) {
|
|
background = bg;
|
|
} else {
|
|
background = COLOR_BLACK;
|
|
}
|
|
return dlg_color_pair(foreground, background);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* End using dialog functions.
|
|
*/
|
|
void
|
|
end_dialog(void)
|
|
{
|
|
if (dialog_state.screen_initialized) {
|
|
dialog_state.screen_initialized = FALSE;
|
|
mouse_close();
|
|
(void) endwin();
|
|
(void) fflush(stdout);
|
|
}
|
|
}
|
|
|
|
#define ESCAPE_LEN 3
|
|
#define isOurEscape(p) (((p)[0] == '\\') && ((p)[1] == 'Z') && ((p)[2] != 0))
|
|
|
|
int
|
|
dlg_count_real_columns(const char *text)
|
|
{
|
|
int result = 0;
|
|
if (*text) {
|
|
result = dlg_count_columns(text);
|
|
if (result && dialog_vars.colors) {
|
|
int hidden = 0;
|
|
while (*text) {
|
|
if (dialog_vars.colors && isOurEscape(text)) {
|
|
hidden += ESCAPE_LEN;
|
|
text += ESCAPE_LEN;
|
|
} else {
|
|
++text;
|
|
}
|
|
}
|
|
result -= hidden;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static int
|
|
centered(int width, const char *string)
|
|
{
|
|
int need = dlg_count_real_columns(string);
|
|
int left;
|
|
|
|
left = (width - need) / 2 - 1;
|
|
if (left < 0)
|
|
left = 0;
|
|
return left;
|
|
}
|
|
|
|
#ifdef USE_WIDE_CURSES
|
|
static bool
|
|
is_combining(const char *txt, int *combined)
|
|
{
|
|
bool result = FALSE;
|
|
|
|
if (*combined == 0) {
|
|
if (UCH(*txt) >= 128) {
|
|
wchar_t wch;
|
|
mbstate_t state;
|
|
size_t given = strlen(txt);
|
|
size_t len;
|
|
|
|
memset(&state, 0, sizeof(state));
|
|
len = mbrtowc(&wch, txt, given, &state);
|
|
if ((int) len > 0 && wcwidth(wch) == 0) {
|
|
*combined = (int) len - 1;
|
|
result = TRUE;
|
|
}
|
|
}
|
|
} else {
|
|
result = TRUE;
|
|
*combined -= 1;
|
|
}
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Print the name (tag) or text from a DIALOG_LISTITEM, highlighting the
|
|
* first character if selected.
|
|
*/
|
|
void
|
|
dlg_print_listitem(WINDOW *win,
|
|
const char *text,
|
|
int climit,
|
|
bool first,
|
|
int selected)
|
|
{
|
|
chtype attr = A_NORMAL;
|
|
int limit;
|
|
const int *cols;
|
|
chtype attrs[4];
|
|
|
|
if (text == 0)
|
|
text = "";
|
|
|
|
if (first) {
|
|
const int *indx = dlg_index_wchars(text);
|
|
attrs[3] = tag_key_selected_attr;
|
|
attrs[2] = tag_key_attr;
|
|
attrs[1] = tag_selected_attr;
|
|
attrs[0] = tag_attr;
|
|
|
|
(void) wattrset(win, selected ? attrs[3] : attrs[2]);
|
|
(void) waddnstr(win, text, indx[1]);
|
|
|
|
if ((int) strlen(text) > indx[1]) {
|
|
limit = dlg_limit_columns(text, climit, 1);
|
|
if (limit > 1) {
|
|
(void) wattrset(win, selected ? attrs[1] : attrs[0]);
|
|
(void) waddnstr(win,
|
|
text + indx[1],
|
|
indx[limit] - indx[1]);
|
|
}
|
|
}
|
|
} else {
|
|
attrs[1] = item_selected_attr;
|
|
attrs[0] = item_attr;
|
|
|
|
cols = dlg_index_columns(text);
|
|
limit = dlg_limit_columns(text, climit, 0);
|
|
|
|
if (limit > 0) {
|
|
(void) wattrset(win, selected ? attrs[1] : attrs[0]);
|
|
dlg_print_text(win, text, cols[limit], &attr);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Print up to 'cols' columns from 'text', optionally rendering our escape
|
|
* sequence for attributes and color.
|
|
*/
|
|
void
|
|
dlg_print_text(WINDOW *win, const char *txt, int cols, chtype *attr)
|
|
{
|
|
int y_origin, x_origin;
|
|
int y_before, x_before = 0;
|
|
int y_after, x_after;
|
|
int tabbed = 0;
|
|
bool thisTab;
|
|
bool ended = FALSE;
|
|
chtype useattr;
|
|
#ifdef USE_WIDE_CURSES
|
|
int combined = 0;
|
|
#endif
|
|
|
|
getyx(win, y_origin, x_origin);
|
|
while (cols > 0 && (*txt != '\0')) {
|
|
if (dialog_vars.colors) {
|
|
while (isOurEscape(txt)) {
|
|
int code;
|
|
|
|
txt += 2;
|
|
switch (code = CharOf(*txt)) {
|
|
#ifdef HAVE_COLOR
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
*attr &= ~A_COLOR;
|
|
*attr |= define_color(win, code - '0');
|
|
break;
|
|
#endif
|
|
case 'B':
|
|
*attr &= ~A_BOLD;
|
|
break;
|
|
case 'b':
|
|
*attr |= A_BOLD;
|
|
break;
|
|
case 'R':
|
|
*attr &= ~A_REVERSE;
|
|
break;
|
|
case 'r':
|
|
*attr |= A_REVERSE;
|
|
break;
|
|
case 'U':
|
|
*attr &= ~A_UNDERLINE;
|
|
break;
|
|
case 'u':
|
|
*attr |= A_UNDERLINE;
|
|
break;
|
|
case 'n':
|
|
*attr = A_NORMAL;
|
|
break;
|
|
}
|
|
++txt;
|
|
}
|
|
}
|
|
if (ended || *txt == '\n' || *txt == '\0')
|
|
break;
|
|
useattr = (*attr) & A_ATTRIBUTES;
|
|
#ifdef HAVE_COLOR
|
|
/*
|
|
* Prevent this from making text invisible when the foreground and
|
|
* background colors happen to be the same, and there's no bold
|
|
* attribute.
|
|
*/
|
|
if ((useattr & A_COLOR) != 0 && (useattr & A_BOLD) == 0) {
|
|
short pair = (short) PAIR_NUMBER(useattr);
|
|
short fg, bg;
|
|
if (pair_content(pair, &fg, &bg) != ERR
|
|
&& fg == bg) {
|
|
useattr &= ~A_COLOR;
|
|
useattr |= dlg_color_pair(fg, ((bg == COLOR_BLACK)
|
|
? COLOR_WHITE
|
|
: COLOR_BLACK));
|
|
}
|
|
}
|
|
#endif
|
|
/*
|
|
* Write the character, using curses to tell exactly how wide it
|
|
* is. If it is a tab, discount that, since the caller thinks
|
|
* tabs are nonprinting, and curses will expand tabs to one or
|
|
* more blanks.
|
|
*/
|
|
thisTab = (CharOf(*txt) == TAB);
|
|
if (thisTab) {
|
|
getyx(win, y_before, x_before);
|
|
(void) y_before;
|
|
}
|
|
(void) waddch(win, CharOf(*txt++) | useattr);
|
|
getyx(win, y_after, x_after);
|
|
if (thisTab && (y_after == y_origin))
|
|
tabbed += (x_after - x_before);
|
|
if ((y_after != y_origin) ||
|
|
(x_after >= (cols + tabbed + x_origin)
|
|
#ifdef USE_WIDE_CURSES
|
|
&& !is_combining(txt, &combined)
|
|
#endif
|
|
)) {
|
|
ended = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Print one line of the prompt in the window within the limits of the
|
|
* specified right margin. The line will end on a word boundary and a pointer
|
|
* to the start of the next line is returned, or a NULL pointer if the end of
|
|
* *prompt is reached.
|
|
*/
|
|
const char *
|
|
dlg_print_line(WINDOW *win,
|
|
chtype *attr,
|
|
const char *prompt,
|
|
int lm, int rm, int *x)
|
|
{
|
|
const char *wrap_ptr;
|
|
const char *test_ptr;
|
|
const char *hide_ptr = 0;
|
|
const int *cols = dlg_index_columns(prompt);
|
|
const int *indx = dlg_index_wchars(prompt);
|
|
int wrap_inx = 0;
|
|
int test_inx = 0;
|
|
int cur_x = lm;
|
|
int hidden = 0;
|
|
int limit = dlg_count_wchars(prompt);
|
|
int n;
|
|
int tabbed = 0;
|
|
|
|
*x = 1;
|
|
|
|
/*
|
|
* Set *test_ptr to the end of the line or the right margin (rm), whichever
|
|
* is less, and set wrap_ptr to the end of the last word in the line.
|
|
*/
|
|
for (n = 0; n < limit; ++n) {
|
|
test_ptr = prompt + indx[test_inx];
|
|
if (*test_ptr == '\n' || *test_ptr == '\0' || cur_x >= (rm + hidden))
|
|
break;
|
|
if (*test_ptr == TAB && n == 0) {
|
|
tabbed = 8; /* workaround for leading tabs */
|
|
} else if (*test_ptr == ' ' && n != 0 && prompt[indx[n - 1]] != ' ') {
|
|
wrap_inx = n;
|
|
*x = cur_x;
|
|
} else if (dialog_vars.colors && isOurEscape(test_ptr)) {
|
|
hide_ptr = test_ptr;
|
|
hidden += ESCAPE_LEN;
|
|
n += (ESCAPE_LEN - 1);
|
|
}
|
|
cur_x = lm + tabbed + cols[n + 1];
|
|
if (cur_x > (rm + hidden))
|
|
break;
|
|
test_inx = n + 1;
|
|
}
|
|
|
|
/*
|
|
* If the line doesn't reach the right margin in the middle of a word, then
|
|
* we don't have to wrap it at the end of the previous word.
|
|
*/
|
|
test_ptr = prompt + indx[test_inx];
|
|
if (*test_ptr == '\n' || *test_ptr == ' ' || *test_ptr == '\0') {
|
|
wrap_inx = test_inx;
|
|
while (wrap_inx > 0 && prompt[indx[wrap_inx - 1]] == ' ') {
|
|
wrap_inx--;
|
|
}
|
|
*x = lm + indx[wrap_inx];
|
|
} else if (*x == 1 && cur_x >= rm) {
|
|
/*
|
|
* If the line has no spaces, then wrap it anyway at the right margin
|
|
*/
|
|
*x = rm;
|
|
wrap_inx = test_inx;
|
|
}
|
|
wrap_ptr = prompt + indx[wrap_inx];
|
|
#ifdef USE_WIDE_CURSES
|
|
if (UCH(*wrap_ptr) >= 128) {
|
|
int combined = 0;
|
|
while (is_combining(wrap_ptr, &combined)) {
|
|
++wrap_ptr;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* If we found hidden text past the last point that we will display,
|
|
* discount that from the displayed length.
|
|
*/
|
|
if ((hide_ptr != 0) && (hide_ptr >= wrap_ptr)) {
|
|
hidden -= ESCAPE_LEN;
|
|
test_ptr = wrap_ptr;
|
|
while (test_ptr < wrap_ptr) {
|
|
if (dialog_vars.colors && isOurEscape(test_ptr)) {
|
|
hidden -= ESCAPE_LEN;
|
|
test_ptr += ESCAPE_LEN;
|
|
} else {
|
|
++test_ptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Print the line if we have a window pointer. Otherwise this routine
|
|
* is just being called for sizing the window.
|
|
*/
|
|
if (win) {
|
|
dlg_print_text(win, prompt, (cols[wrap_inx] - hidden), attr);
|
|
}
|
|
|
|
/* *x tells the calling function how long the line was */
|
|
if (*x == 1)
|
|
*x = rm;
|
|
|
|
*x -= hidden;
|
|
|
|
/* Find the start of the next line and return a pointer to it */
|
|
test_ptr = wrap_ptr;
|
|
while (*test_ptr == ' ')
|
|
test_ptr++;
|
|
if (*test_ptr == '\n')
|
|
test_ptr++;
|
|
dlg_finish_string(prompt);
|
|
return (test_ptr);
|
|
}
|
|
|
|
static void
|
|
justify_text(WINDOW *win,
|
|
const char *prompt,
|
|
int limit_y,
|
|
int limit_x,
|
|
int *high, int *wide)
|
|
{
|
|
chtype attr = A_NORMAL;
|
|
int x = (2 * MARGIN);
|
|
int y = MARGIN;
|
|
int max_x = 2;
|
|
int lm = (2 * MARGIN); /* left margin (box-border plus a space) */
|
|
int rm = limit_x; /* right margin */
|
|
int bm = limit_y; /* bottom margin */
|
|
int last_y = 0, last_x = 0;
|
|
|
|
if (win) {
|
|
rm -= (2 * MARGIN);
|
|
bm -= (2 * MARGIN);
|
|
}
|
|
if (prompt == 0)
|
|
prompt = "";
|
|
|
|
if (win != 0)
|
|
getyx(win, last_y, last_x);
|
|
while (y <= bm && *prompt) {
|
|
x = lm;
|
|
|
|
if (*prompt == '\n') {
|
|
while (*prompt == '\n' && y < bm) {
|
|
if (*(prompt + 1) != '\0') {
|
|
++y;
|
|
if (win != 0)
|
|
(void) wmove(win, y, lm);
|
|
}
|
|
prompt++;
|
|
}
|
|
} else if (win != 0)
|
|
(void) wmove(win, y, lm);
|
|
|
|
if (*prompt) {
|
|
prompt = dlg_print_line(win, &attr, prompt, lm, rm, &x);
|
|
if (win != 0)
|
|
getyx(win, last_y, last_x);
|
|
}
|
|
if (*prompt) {
|
|
++y;
|
|
if (win != 0)
|
|
(void) wmove(win, y, lm);
|
|
}
|
|
max_x = MAX(max_x, x);
|
|
}
|
|
/* Move back to the last position after drawing prompt, for msgbox. */
|
|
if (win != 0)
|
|
(void) wmove(win, last_y, last_x);
|
|
|
|
/* Set the final height and width for the calling function */
|
|
if (high != 0)
|
|
*high = y;
|
|
if (wide != 0)
|
|
*wide = max_x;
|
|
}
|
|
|
|
/*
|
|
* Print a string of text in a window, automatically wrap around to the next
|
|
* line if the string is too long to fit on one line. Note that the string may
|
|
* contain embedded newlines.
|
|
*/
|
|
void
|
|
dlg_print_autowrap(WINDOW *win, const char *prompt, int height, int width)
|
|
{
|
|
justify_text(win, prompt,
|
|
height,
|
|
width,
|
|
(int *) 0, (int *) 0);
|
|
}
|
|
|
|
/*
|
|
* Display the message in a scrollable window. Actually the way it works is
|
|
* that we create a "tall" window of the proper width, let the text wrap within
|
|
* that, and copy a slice of the result to the dialog.
|
|
*
|
|
* It works for ncurses. Other curses implementations show only blanks (Tru64)
|
|
* or garbage (NetBSD).
|
|
*/
|
|
int
|
|
dlg_print_scrolled(WINDOW *win,
|
|
const char *prompt,
|
|
int offset,
|
|
int height,
|
|
int width,
|
|
int pauseopt)
|
|
{
|
|
int oldy, oldx;
|
|
int last = 0;
|
|
|
|
(void) pauseopt; /* used only for ncurses */
|
|
|
|
getyx(win, oldy, oldx);
|
|
#ifdef NCURSES_VERSION
|
|
if (pauseopt) {
|
|
int wide = width - (2 * MARGIN);
|
|
int high = LINES;
|
|
int y, x;
|
|
int len;
|
|
int percent;
|
|
WINDOW *dummy;
|
|
char buffer[5];
|
|
|
|
#if defined(NCURSES_VERSION_PATCH) && NCURSES_VERSION_PATCH >= 20040417
|
|
/*
|
|
* If we're not limited by the screensize, allow text to possibly be
|
|
* one character per line.
|
|
*/
|
|
if ((len = dlg_count_columns(prompt)) > high)
|
|
high = len;
|
|
#endif
|
|
dummy = newwin(high, width, 0, 0);
|
|
if (dummy == 0) {
|
|
(void) wattrset(win, dialog_attr);
|
|
dlg_print_autowrap(win, prompt, height + 1 + (3 * MARGIN), width);
|
|
last = 0;
|
|
} else {
|
|
wbkgdset(dummy, dialog_attr | ' ');
|
|
(void) wattrset(dummy, dialog_attr);
|
|
werase(dummy);
|
|
dlg_print_autowrap(dummy, prompt, high, width);
|
|
getyx(dummy, y, x);
|
|
(void) x;
|
|
|
|
copywin(dummy, /* srcwin */
|
|
win, /* dstwin */
|
|
offset + MARGIN, /* sminrow */
|
|
MARGIN, /* smincol */
|
|
MARGIN, /* dminrow */
|
|
MARGIN, /* dmincol */
|
|
height, /* dmaxrow */
|
|
wide, /* dmaxcol */
|
|
FALSE);
|
|
|
|
delwin(dummy);
|
|
|
|
/* if the text is incomplete, or we have scrolled, show the percentage */
|
|
if (y > 0 && wide > 4) {
|
|
percent = (int) ((height + offset) * 100.0 / y);
|
|
if (percent < 0)
|
|
percent = 0;
|
|
if (percent > 100)
|
|
percent = 100;
|
|
if (offset != 0 || percent != 100) {
|
|
(void) wattrset(win, position_indicator_attr);
|
|
(void) wmove(win, MARGIN + height, wide - 4);
|
|
(void) sprintf(buffer, "%d%%", percent);
|
|
(void) waddstr(win, buffer);
|
|
if ((len = (int) strlen(buffer)) < 4) {
|
|
(void) wattrset(win, border_attr);
|
|
whline(win, dlg_boxchar(ACS_HLINE), 4 - len);
|
|
}
|
|
}
|
|
}
|
|
last = (y - height);
|
|
}
|
|
} else
|
|
#endif
|
|
{
|
|
(void) offset;
|
|
(void) wattrset(win, dialog_attr);
|
|
dlg_print_autowrap(win, prompt, height + 1 + (3 * MARGIN), width);
|
|
last = 0;
|
|
}
|
|
wmove(win, oldy, oldx);
|
|
return last;
|
|
}
|
|
|
|
int
|
|
dlg_check_scrolled(int key, int last, int page, bool * show, int *offset)
|
|
{
|
|
int code = 0;
|
|
|
|
*show = FALSE;
|
|
|
|
switch (key) {
|
|
case DLGK_PAGE_FIRST:
|
|
if (*offset > 0) {
|
|
*offset = 0;
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
case DLGK_PAGE_LAST:
|
|
if (*offset < last) {
|
|
*offset = last;
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
case DLGK_GRID_UP:
|
|
if (*offset > 0) {
|
|
--(*offset);
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
case DLGK_GRID_DOWN:
|
|
if (*offset < last) {
|
|
++(*offset);
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
case DLGK_PAGE_PREV:
|
|
if (*offset > 0) {
|
|
*offset -= page;
|
|
if (*offset < 0)
|
|
*offset = 0;
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
case DLGK_PAGE_NEXT:
|
|
if (*offset < last) {
|
|
*offset += page;
|
|
if (*offset > last)
|
|
*offset = last;
|
|
*show = TRUE;
|
|
}
|
|
break;
|
|
default:
|
|
code = -1;
|
|
break;
|
|
}
|
|
return code;
|
|
}
|
|
|
|
/*
|
|
* Calculate the window size for preformatted text. This will calculate box
|
|
* dimensions that are at or close to the specified aspect ratio for the prompt
|
|
* string with all spaces and newlines preserved and additional newlines added
|
|
* as necessary.
|
|
*/
|
|
static void
|
|
auto_size_preformatted(const char *prompt, int *height, int *width)
|
|
{
|
|
int high = 0, wide = 0;
|
|
float car; /* Calculated Aspect Ratio */
|
|
float diff;
|
|
int max_y = SLINES - 1;
|
|
int max_x = SCOLS - 2;
|
|
int max_width = max_x;
|
|
int ar = dialog_state.aspect_ratio;
|
|
|
|
/* Get the initial dimensions */
|
|
justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
|
|
car = (float) (wide / high);
|
|
|
|
/*
|
|
* If the aspect ratio is greater than it should be, then decrease the
|
|
* width proportionately.
|
|
*/
|
|
if (car > ar) {
|
|
diff = car / (float) ar;
|
|
max_x = (int) ((float) wide / diff + 4);
|
|
justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
|
|
car = (float) wide / (float) high;
|
|
}
|
|
|
|
/*
|
|
* If the aspect ratio is too small after decreasing the width, then
|
|
* incrementally increase the width until the aspect ratio is equal to or
|
|
* greater than the specified aspect ratio.
|
|
*/
|
|
while (car < ar && max_x < max_width) {
|
|
max_x += 4;
|
|
justify_text((WINDOW *) 0, prompt, max_y, max_x, &high, &wide);
|
|
car = (float) (wide / high);
|
|
}
|
|
|
|
*height = high;
|
|
*width = wide;
|
|
}
|
|
|
|
/*
|
|
* Find the length of the longest "word" in the given string. By setting the
|
|
* widget width at least this long, we can avoid splitting a word on the
|
|
* margin.
|
|
*/
|
|
static int
|
|
longest_word(const char *string)
|
|
{
|
|
int length, result = 0;
|
|
|
|
while (*string != '\0') {
|
|
length = 0;
|
|
while (*string != '\0' && !isspace(UCH(*string))) {
|
|
length++;
|
|
string++;
|
|
}
|
|
result = MAX(result, length);
|
|
if (*string != '\0')
|
|
string++;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* if (height or width == -1) Maximize()
|
|
* if (height or width == 0), justify and return actual limits.
|
|
*/
|
|
static void
|
|
real_auto_size(const char *title,
|
|
const char *prompt,
|
|
int *height, int *width,
|
|
int boxlines, int mincols)
|
|
{
|
|
int x = (dialog_vars.begin_set ? dialog_vars.begin_x : 2);
|
|
int y = (dialog_vars.begin_set ? dialog_vars.begin_y : 1);
|
|
int title_length = title ? dlg_count_columns(title) : 0;
|
|
int nc = 4;
|
|
int high;
|
|
int wide;
|
|
int save_high = *height;
|
|
int save_wide = *width;
|
|
|
|
if (prompt == 0) {
|
|
if (*height == 0)
|
|
*height = -1;
|
|
if (*width == 0)
|
|
*width = -1;
|
|
}
|
|
|
|
if (*height > 0) {
|
|
high = *height;
|
|
} else {
|
|
high = SLINES - y;
|
|
}
|
|
|
|
if (*width <= 0) {
|
|
if (prompt != 0) {
|
|
wide = MAX(title_length, mincols);
|
|
if (strchr(prompt, '\n') == 0) {
|
|
double val = (dialog_state.aspect_ratio *
|
|
dlg_count_real_columns(prompt));
|
|
double xxx = sqrt(val);
|
|
int tmp = (int) xxx;
|
|
wide = MAX(wide, tmp);
|
|
wide = MAX(wide, longest_word(prompt));
|
|
justify_text((WINDOW *) 0, prompt, high, wide, height, width);
|
|
} else {
|
|
auto_size_preformatted(prompt, height, width);
|
|
}
|
|
} else {
|
|
wide = SCOLS - x;
|
|
justify_text((WINDOW *) 0, prompt, high, wide, height, width);
|
|
}
|
|
}
|
|
|
|
if (*width < title_length) {
|
|
justify_text((WINDOW *) 0, prompt, high, title_length, height, width);
|
|
*width = title_length;
|
|
}
|
|
|
|
if (*width < mincols && save_wide == 0)
|
|
*width = mincols;
|
|
if (prompt != 0) {
|
|
*width += nc;
|
|
*height += boxlines + 2;
|
|
}
|
|
if (save_high > 0)
|
|
*height = save_high;
|
|
if (save_wide > 0)
|
|
*width = save_wide;
|
|
}
|
|
|
|
/* End of real_auto_size() */
|
|
|
|
void
|
|
dlg_auto_size(const char *title,
|
|
const char *prompt,
|
|
int *height,
|
|
int *width,
|
|
int boxlines,
|
|
int mincols)
|
|
{
|
|
real_auto_size(title, prompt, height, width, boxlines, mincols);
|
|
|
|
if (*width > SCOLS) {
|
|
(*height)++;
|
|
*width = SCOLS;
|
|
}
|
|
|
|
if (*height > SLINES)
|
|
*height = SLINES;
|
|
}
|
|
|
|
/*
|
|
* if (height or width == -1) Maximize()
|
|
* if (height or width == 0)
|
|
* height=MIN(SLINES, num.lines in fd+n);
|
|
* width=MIN(SCOLS, MAX(longer line+n, mincols));
|
|
*/
|
|
void
|
|
dlg_auto_sizefile(const char *title,
|
|
const char *file,
|
|
int *height,
|
|
int *width,
|
|
int boxlines,
|
|
int mincols)
|
|
{
|
|
int count = 0;
|
|
int len = title ? dlg_count_columns(title) : 0;
|
|
int nc = 4;
|
|
int numlines = 2;
|
|
long offset;
|
|
int ch;
|
|
FILE *fd;
|
|
|
|
/* Open input file for reading */
|
|
if ((fd = fopen(file, "rb")) == NULL)
|
|
dlg_exiterr("dlg_auto_sizefile: Cannot open input file %s", file);
|
|
|
|
if ((*height == -1) || (*width == -1)) {
|
|
*height = SLINES - (dialog_vars.begin_set ? dialog_vars.begin_y : 0);
|
|
*width = SCOLS - (dialog_vars.begin_set ? dialog_vars.begin_x : 0);
|
|
}
|
|
if ((*height != 0) && (*width != 0)) {
|
|
(void) fclose(fd);
|
|
if (*width > SCOLS)
|
|
*width = SCOLS;
|
|
if (*height > SLINES)
|
|
*height = SLINES;
|
|
return;
|
|
}
|
|
|
|
while (!feof(fd)) {
|
|
offset = 0;
|
|
while (((ch = getc(fd)) != '\n') && !feof(fd))
|
|
if ((ch == TAB) && (dialog_vars.tab_correct))
|
|
offset += dialog_state.tab_len - (offset % dialog_state.tab_len);
|
|
else
|
|
offset++;
|
|
|
|
if (offset > len)
|
|
len = (int) offset;
|
|
|
|
count++;
|
|
}
|
|
|
|
/* now 'count' has the number of lines of fd and 'len' the max length */
|
|
|
|
*height = MIN(SLINES, count + numlines + boxlines);
|
|
*width = MIN(SCOLS, MAX((len + nc), mincols));
|
|
/* here width and height can be maximized if > SCOLS|SLINES because
|
|
textbox-like widgets don't put all <file> on the screen.
|
|
Msgbox-like widget instead have to put all <text> correctly. */
|
|
|
|
(void) fclose(fd);
|
|
}
|
|
|
|
static chtype
|
|
dlg_get_cell_attrs(WINDOW *win)
|
|
{
|
|
chtype result;
|
|
#ifdef USE_WIDE_CURSES
|
|
cchar_t wch;
|
|
wchar_t cc;
|
|
attr_t attrs;
|
|
short pair;
|
|
if (win_wch(win, &wch) == OK
|
|
&& getcchar(&wch, &cc, &attrs, &pair, NULL) == OK) {
|
|
result = attrs;
|
|
} else {
|
|
result = 0;
|
|
}
|
|
#else
|
|
result = winch(win) & (A_ATTRIBUTES & ~A_COLOR);
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Draw a rectangular box with line drawing characters.
|
|
*
|
|
* borderchar is used to color the upper/left edges.
|
|
*
|
|
* boxchar is used to color the right/lower edges. It also is fill-color used
|
|
* for the box contents.
|
|
*
|
|
* Normally, if you are drawing a scrollable box, use menubox_border_attr for
|
|
* boxchar, and menubox_attr for borderchar since the scroll-arrows are drawn
|
|
* with menubox_attr at the top, and menubox_border_attr at the bottom. That
|
|
* also (given the default color choices) produces a recessed effect.
|
|
*
|
|
* If you want a raised effect (and are not going to use the scroll-arrows),
|
|
* reverse this choice.
|
|
*/
|
|
void
|
|
dlg_draw_box2(WINDOW *win, int y, int x, int height, int width,
|
|
chtype boxchar, chtype borderchar, chtype borderchar2)
|
|
{
|
|
int i, j;
|
|
chtype save = dlg_get_attrs(win);
|
|
|
|
(void) wattrset(win, 0);
|
|
for (i = 0; i < height; i++) {
|
|
(void) wmove(win, y + i, x);
|
|
for (j = 0; j < width; j++)
|
|
if (!i && !j)
|
|
(void) waddch(win, borderchar | dlg_boxchar(ACS_ULCORNER));
|
|
else if (i == height - 1 && !j)
|
|
(void) waddch(win, borderchar | dlg_boxchar(ACS_LLCORNER));
|
|
else if (!i && j == width - 1)
|
|
(void) waddch(win, borderchar2 | dlg_boxchar(ACS_URCORNER));
|
|
else if (i == height - 1 && j == width - 1)
|
|
(void) waddch(win, borderchar2 | dlg_boxchar(ACS_LRCORNER));
|
|
else if (!i)
|
|
(void) waddch(win, borderchar | dlg_boxchar(ACS_HLINE));
|
|
else if (i == height - 1)
|
|
(void) waddch(win, borderchar2 | dlg_boxchar(ACS_HLINE));
|
|
else if (!j)
|
|
(void) waddch(win, borderchar | dlg_boxchar(ACS_VLINE));
|
|
else if (j == width - 1)
|
|
(void) waddch(win, borderchar2 | dlg_boxchar(ACS_VLINE));
|
|
else
|
|
(void) waddch(win, boxchar | ' ');
|
|
}
|
|
(void) wattrset(win, save);
|
|
}
|
|
|
|
void
|
|
dlg_draw_box(WINDOW *win, int y, int x, int height, int width,
|
|
chtype boxchar, chtype borderchar)
|
|
{
|
|
dlg_draw_box2(win, y, x, height, width, boxchar, borderchar, boxchar);
|
|
}
|
|
|
|
static DIALOG_WINDOWS *
|
|
find_window(WINDOW *win)
|
|
{
|
|
DIALOG_WINDOWS *result = 0;
|
|
DIALOG_WINDOWS *p;
|
|
|
|
for (p = dialog_state.all_windows; p != 0; p = p->next) {
|
|
if (p->normal == win) {
|
|
result = p;
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#ifdef HAVE_COLOR
|
|
/*
|
|
* If we have wchgat(), use that for updating shadow attributes, to work with
|
|
* wide-character data.
|
|
*/
|
|
|
|
/*
|
|
* Check if the given point is "in" the given window. If so, return the window
|
|
* pointer, otherwise null.
|
|
*/
|
|
static WINDOW *
|
|
in_window(WINDOW *win, int y, int x)
|
|
{
|
|
WINDOW *result = 0;
|
|
int y_base = getbegy(win);
|
|
int x_base = getbegx(win);
|
|
int y_last = getmaxy(win) + y_base;
|
|
int x_last = getmaxx(win) + x_base;
|
|
|
|
if (y >= y_base && y <= y_last && x >= x_base && x <= x_last)
|
|
result = win;
|
|
return result;
|
|
}
|
|
|
|
static WINDOW *
|
|
window_at_cell(DIALOG_WINDOWS * dw, int y, int x)
|
|
{
|
|
WINDOW *result = 0;
|
|
DIALOG_WINDOWS *p;
|
|
int y_want = y + getbegy(dw->shadow);
|
|
int x_want = x + getbegx(dw->shadow);
|
|
|
|
for (p = dialog_state.all_windows; p != 0; p = p->next) {
|
|
if (dw->normal != p->normal
|
|
&& dw->shadow != p->normal
|
|
&& (result = in_window(p->normal, y_want, x_want)) != 0) {
|
|
break;
|
|
}
|
|
}
|
|
if (result == 0) {
|
|
result = stdscr;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static bool
|
|
in_shadow(WINDOW *normal, WINDOW *shadow, int y, int x)
|
|
{
|
|
bool result = FALSE;
|
|
int ybase = getbegy(normal);
|
|
int ylast = getmaxy(normal) + ybase;
|
|
int xbase = getbegx(normal);
|
|
int xlast = getmaxx(normal) + xbase;
|
|
|
|
y += getbegy(shadow);
|
|
x += getbegx(shadow);
|
|
|
|
if (y >= ybase + SHADOW_ROWS
|
|
&& y < ylast + SHADOW_ROWS
|
|
&& x >= xlast
|
|
&& x < xlast + SHADOW_COLS) {
|
|
/* in the right-side */
|
|
result = TRUE;
|
|
} else if (y >= ylast
|
|
&& y < ylast + SHADOW_ROWS
|
|
&& x >= ybase + SHADOW_COLS
|
|
&& x < ylast + SHADOW_COLS) {
|
|
/* check the bottom */
|
|
result = TRUE;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* When erasing a shadow, check each cell to make sure that it is not part of
|
|
* another box's shadow. This is a little complicated since most shadows are
|
|
* merged onto stdscr.
|
|
*/
|
|
static bool
|
|
last_shadow(DIALOG_WINDOWS * dw, int y, int x)
|
|
{
|
|
DIALOG_WINDOWS *p;
|
|
bool result = TRUE;
|
|
|
|
for (p = dialog_state.all_windows; p != 0; p = p->next) {
|
|
if (p->normal != dw->normal
|
|
&& in_shadow(p->normal, dw->shadow, y, x)) {
|
|
result = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
repaint_cell(DIALOG_WINDOWS * dw, bool draw, int y, int x)
|
|
{
|
|
WINDOW *win = dw->shadow;
|
|
WINDOW *cellwin;
|
|
int y2, x2;
|
|
|
|
if ((cellwin = window_at_cell(dw, y, x)) != 0
|
|
&& (draw || last_shadow(dw, y, x))
|
|
&& (y2 = (y + getbegy(win) - getbegy(cellwin))) >= 0
|
|
&& (x2 = (x + getbegx(win) - getbegx(cellwin))) >= 0
|
|
&& wmove(cellwin, y2, x2) != ERR) {
|
|
chtype the_cell = dlg_get_attrs(cellwin);
|
|
chtype the_attr = (draw ? shadow_attr : the_cell);
|
|
|
|
if (dlg_get_cell_attrs(cellwin) & A_ALTCHARSET) {
|
|
the_attr |= A_ALTCHARSET;
|
|
}
|
|
#if USE_WCHGAT
|
|
wchgat(cellwin, 1,
|
|
the_attr & (chtype) (~A_COLOR),
|
|
(short) PAIR_NUMBER(the_attr),
|
|
NULL);
|
|
#else
|
|
{
|
|
chtype the_char = ((winch(cellwin) & A_CHARTEXT) | the_attr);
|
|
(void) waddch(cellwin, the_char);
|
|
}
|
|
#endif
|
|
wnoutrefresh(cellwin);
|
|
}
|
|
}
|
|
|
|
#define RepaintCell(dw, draw, y, x) repaint_cell(dw, draw, y, x)
|
|
|
|
static void
|
|
repaint_shadow(DIALOG_WINDOWS * dw, bool draw, int y, int x, int height, int width)
|
|
{
|
|
int i, j;
|
|
|
|
if (UseShadow(dw)) {
|
|
#if !USE_WCHGAT
|
|
chtype save = dlg_get_attrs(dw->shadow);
|
|
(void) wattrset(dw->shadow, draw ? shadow_attr : screen_attr);
|
|
#endif
|
|
for (i = 0; i < SHADOW_ROWS; ++i) {
|
|
for (j = 0; j < width; ++j) {
|
|
RepaintCell(dw, draw, i + y + height, j + x + SHADOW_COLS);
|
|
}
|
|
}
|
|
for (i = 0; i < height; i++) {
|
|
for (j = 0; j < SHADOW_COLS; ++j) {
|
|
RepaintCell(dw, draw, i + y + SHADOW_ROWS, j + x + width);
|
|
}
|
|
}
|
|
(void) wnoutrefresh(dw->shadow);
|
|
#if !USE_WCHGAT
|
|
(void) wattrset(dw->shadow, save);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Draw a shadow on the parent window corresponding to the right- and
|
|
* bottom-edge of the child window, to give a 3-dimensional look.
|
|
*/
|
|
static void
|
|
draw_childs_shadow(DIALOG_WINDOWS * dw)
|
|
{
|
|
if (UseShadow(dw)) {
|
|
repaint_shadow(dw,
|
|
TRUE,
|
|
getbegy(dw->normal) - getbegy(dw->shadow),
|
|
getbegx(dw->normal) - getbegx(dw->shadow),
|
|
getmaxy(dw->normal),
|
|
getmaxx(dw->normal));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Erase a shadow on the parent window corresponding to the right- and
|
|
* bottom-edge of the child window.
|
|
*/
|
|
static void
|
|
erase_childs_shadow(DIALOG_WINDOWS * dw)
|
|
{
|
|
if (UseShadow(dw)) {
|
|
repaint_shadow(dw,
|
|
FALSE,
|
|
getbegy(dw->normal) - getbegy(dw->shadow),
|
|
getbegx(dw->normal) - getbegx(dw->shadow),
|
|
getmaxy(dw->normal),
|
|
getmaxx(dw->normal));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Draw shadows along the right and bottom edge to give a more 3D look
|
|
* to the boxes.
|
|
*/
|
|
void
|
|
dlg_draw_shadow(WINDOW *win, int y, int x, int height, int width)
|
|
{
|
|
repaint_shadow(find_window(win), TRUE, y, x, height, width);
|
|
}
|
|
#endif /* HAVE_COLOR */
|
|
|
|
/*
|
|
* Allow shell scripts to remap the exit codes so they can distinguish ESC
|
|
* from ERROR.
|
|
*/
|
|
void
|
|
dlg_exit(int code)
|
|
{
|
|
/* *INDENT-OFF* */
|
|
static const struct {
|
|
int code;
|
|
const char *name;
|
|
} table[] = {
|
|
{ DLG_EXIT_CANCEL, "DIALOG_CANCEL" },
|
|
{ DLG_EXIT_ERROR, "DIALOG_ERROR" },
|
|
{ DLG_EXIT_ESC, "DIALOG_ESC" },
|
|
{ DLG_EXIT_EXTRA, "DIALOG_EXTRA" },
|
|
{ DLG_EXIT_HELP, "DIALOG_HELP" },
|
|
{ DLG_EXIT_OK, "DIALOG_OK" },
|
|
{ DLG_EXIT_ITEM_HELP, "DIALOG_ITEM_HELP" },
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
unsigned n;
|
|
char *name;
|
|
char *temp;
|
|
long value;
|
|
bool overridden = FALSE;
|
|
|
|
retry:
|
|
for (n = 0; n < sizeof(table) / sizeof(table[0]); n++) {
|
|
if (table[n].code == code) {
|
|
if ((name = getenv(table[n].name)) != 0) {
|
|
value = strtol(name, &temp, 0);
|
|
if (temp != 0 && temp != name && *temp == '\0') {
|
|
code = (int) value;
|
|
overridden = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Prior to 2004/12/19, a widget using --item-help would exit with "OK"
|
|
* if the help button were selected. Now we want to exit with "HELP",
|
|
* but allow the environment variable to override.
|
|
*/
|
|
if (code == DLG_EXIT_ITEM_HELP && !overridden) {
|
|
code = DLG_EXIT_HELP;
|
|
goto retry;
|
|
}
|
|
#ifdef HAVE_DLG_TRACE
|
|
dlg_trace((const char *) 0); /* close it */
|
|
#endif
|
|
|
|
#ifdef NO_LEAKS
|
|
_dlg_inputstr_leaks();
|
|
#if defined(NCURSES_VERSION) && defined(HAVE__NC_FREE_AND_EXIT)
|
|
_nc_free_and_exit(code);
|
|
#endif
|
|
#endif
|
|
|
|
if (dialog_state.input == stdin) {
|
|
exit(code);
|
|
} else {
|
|
/*
|
|
* Just in case of using --input-fd option, do not
|
|
* call atexit functions of ncurses which may hang.
|
|
*/
|
|
if (dialog_state.input) {
|
|
fclose(dialog_state.input);
|
|
dialog_state.input = 0;
|
|
}
|
|
if (dialog_state.pipe_input) {
|
|
if (dialog_state.pipe_input != stdin) {
|
|
fclose(dialog_state.pipe_input);
|
|
dialog_state.pipe_input = 0;
|
|
}
|
|
}
|
|
_exit(code);
|
|
}
|
|
}
|
|
|
|
/* quit program killing all tailbg */
|
|
void
|
|
dlg_exiterr(const char *fmt,...)
|
|
{
|
|
int retval;
|
|
va_list ap;
|
|
|
|
end_dialog();
|
|
|
|
(void) fputc('\n', stderr);
|
|
va_start(ap, fmt);
|
|
(void) vfprintf(stderr, fmt, ap);
|
|
va_end(ap);
|
|
(void) fputc('\n', stderr);
|
|
|
|
dlg_killall_bg(&retval);
|
|
|
|
(void) fflush(stderr);
|
|
(void) fflush(stdout);
|
|
dlg_exit(DLG_EXIT_ERROR);
|
|
}
|
|
|
|
void
|
|
dlg_beeping(void)
|
|
{
|
|
if (dialog_vars.beep_signal) {
|
|
(void) beep();
|
|
dialog_vars.beep_signal = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
dlg_print_size(int height, int width)
|
|
{
|
|
if (dialog_vars.print_siz)
|
|
fprintf(dialog_state.output, "Size: %d, %d\n", height, width);
|
|
}
|
|
|
|
void
|
|
dlg_ctl_size(int height, int width)
|
|
{
|
|
if (dialog_vars.size_err) {
|
|
if ((width > COLS) || (height > LINES)) {
|
|
dlg_exiterr("Window too big. (height, width) = (%d, %d). Max allowed (%d, %d).",
|
|
height, width, LINES, COLS);
|
|
}
|
|
#ifdef HAVE_COLOR
|
|
else if ((dialog_state.use_shadow)
|
|
&& ((width > SCOLS || height > SLINES))) {
|
|
if ((width <= COLS) && (height <= LINES)) {
|
|
/* try again, without shadows */
|
|
dialog_state.use_shadow = 0;
|
|
} else {
|
|
dlg_exiterr("Window+Shadow too big. (height, width) = (%d, %d). Max allowed (%d, %d).",
|
|
height, width, SLINES, SCOLS);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If the --tab-correct was not selected, convert tabs to single spaces.
|
|
*/
|
|
void
|
|
dlg_tab_correct_str(char *prompt)
|
|
{
|
|
char *ptr;
|
|
|
|
if (dialog_vars.tab_correct) {
|
|
while ((ptr = strchr(prompt, TAB)) != NULL) {
|
|
*ptr = ' ';
|
|
prompt = ptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
dlg_calc_listh(int *height, int *list_height, int item_no)
|
|
{
|
|
/* calculate new height and list_height */
|
|
int rows = SLINES - (dialog_vars.begin_set ? dialog_vars.begin_y : 0);
|
|
if (rows - (*height) > 0) {
|
|
if (rows - (*height) > item_no)
|
|
*list_height = item_no;
|
|
else
|
|
*list_height = rows - (*height);
|
|
}
|
|
(*height) += (*list_height);
|
|
}
|
|
|
|
/* obsolete */
|
|
int
|
|
dlg_calc_listw(int item_no, char **items, int group)
|
|
{
|
|
int n, i, len1 = 0, len2 = 0;
|
|
for (i = 0; i < (item_no * group); i += group) {
|
|
if ((n = dlg_count_columns(items[i])) > len1)
|
|
len1 = n;
|
|
if ((n = dlg_count_columns(items[i + 1])) > len2)
|
|
len2 = n;
|
|
}
|
|
return len1 + len2;
|
|
}
|
|
|
|
int
|
|
dlg_calc_list_width(int item_no, DIALOG_LISTITEM * items)
|
|
{
|
|
int n, i, len1 = 0, len2 = 0;
|
|
int bits = ((dialog_vars.no_tags ? 1 : 0)
|
|
+ (dialog_vars.no_items ? 2 : 0));
|
|
|
|
for (i = 0; i < item_no; ++i) {
|
|
switch (bits) {
|
|
case 0:
|
|
/* FALLTHRU */
|
|
case 1:
|
|
if ((n = dlg_count_columns(items[i].name)) > len1)
|
|
len1 = n;
|
|
if ((n = dlg_count_columns(items[i].text)) > len2)
|
|
len2 = n;
|
|
break;
|
|
case 2:
|
|
/* FALLTHRU */
|
|
case 3:
|
|
if ((n = dlg_count_columns(items[i].name)) > len1)
|
|
len1 = n;
|
|
break;
|
|
}
|
|
}
|
|
return len1 + len2;
|
|
}
|
|
|
|
char *
|
|
dlg_strempty(void)
|
|
{
|
|
static char empty[] = "";
|
|
return empty;
|
|
}
|
|
|
|
char *
|
|
dlg_strclone(const char *cprompt)
|
|
{
|
|
char *prompt = dlg_malloc(char, strlen(cprompt) + 1);
|
|
assert_ptr(prompt, "dlg_strclone");
|
|
strcpy(prompt, cprompt);
|
|
return prompt;
|
|
}
|
|
|
|
chtype
|
|
dlg_asciibox(chtype ch)
|
|
{
|
|
chtype result = 0;
|
|
|
|
if (ch == ACS_ULCORNER)
|
|
result = '+';
|
|
else if (ch == ACS_LLCORNER)
|
|
result = '+';
|
|
else if (ch == ACS_URCORNER)
|
|
result = '+';
|
|
else if (ch == ACS_LRCORNER)
|
|
result = '+';
|
|
else if (ch == ACS_HLINE)
|
|
result = '-';
|
|
else if (ch == ACS_VLINE)
|
|
result = '|';
|
|
else if (ch == ACS_LTEE)
|
|
result = '+';
|
|
else if (ch == ACS_RTEE)
|
|
result = '+';
|
|
else if (ch == ACS_UARROW)
|
|
result = '^';
|
|
else if (ch == ACS_DARROW)
|
|
result = 'v';
|
|
|
|
return result;
|
|
}
|
|
|
|
chtype
|
|
dlg_boxchar(chtype ch)
|
|
{
|
|
chtype result = dlg_asciibox(ch);
|
|
|
|
if (result != 0) {
|
|
if (dialog_vars.ascii_lines)
|
|
ch = result;
|
|
else if (dialog_vars.no_lines)
|
|
ch = ' ';
|
|
}
|
|
return ch;
|
|
}
|
|
|
|
int
|
|
dlg_box_x_ordinate(int width)
|
|
{
|
|
int x;
|
|
|
|
if (dialog_vars.begin_set == 1) {
|
|
x = dialog_vars.begin_x;
|
|
} else {
|
|
/* center dialog box on screen unless --begin-set */
|
|
x = (SCOLS - width) / 2;
|
|
}
|
|
return x;
|
|
}
|
|
|
|
int
|
|
dlg_box_y_ordinate(int height)
|
|
{
|
|
int y;
|
|
|
|
if (dialog_vars.begin_set == 1) {
|
|
y = dialog_vars.begin_y;
|
|
} else {
|
|
/* center dialog box on screen unless --begin-set */
|
|
y = (SLINES - height) / 2;
|
|
}
|
|
return y;
|
|
}
|
|
|
|
void
|
|
dlg_draw_title(WINDOW *win, const char *title)
|
|
{
|
|
if (title != NULL) {
|
|
chtype attr = A_NORMAL;
|
|
chtype save = dlg_get_attrs(win);
|
|
int x = centered(getmaxx(win), title);
|
|
|
|
(void) wattrset(win, title_attr);
|
|
wmove(win, 0, x);
|
|
dlg_print_text(win, title, getmaxx(win) - x, &attr);
|
|
(void) wattrset(win, save);
|
|
dlg_finish_string(title);
|
|
}
|
|
}
|
|
|
|
void
|
|
dlg_draw_bottom_box2(WINDOW *win, chtype on_left, chtype on_right, chtype on_inside)
|
|
{
|
|
int width = getmaxx(win);
|
|
int height = getmaxy(win);
|
|
int i;
|
|
|
|
(void) wattrset(win, on_left);
|
|
(void) wmove(win, height - 3, 0);
|
|
(void) waddch(win, dlg_boxchar(ACS_LTEE));
|
|
for (i = 0; i < width - 2; i++)
|
|
(void) waddch(win, dlg_boxchar(ACS_HLINE));
|
|
(void) wattrset(win, on_right);
|
|
(void) waddch(win, dlg_boxchar(ACS_RTEE));
|
|
(void) wattrset(win, on_inside);
|
|
(void) wmove(win, height - 2, 1);
|
|
for (i = 0; i < width - 2; i++)
|
|
(void) waddch(win, ' ');
|
|
}
|
|
|
|
void
|
|
dlg_draw_bottom_box(WINDOW *win)
|
|
{
|
|
dlg_draw_bottom_box2(win, border_attr, dialog_attr, dialog_attr);
|
|
}
|
|
|
|
/*
|
|
* Remove a window, repainting everything else. This would be simpler if we
|
|
* used the panel library, but that is not _always_ available.
|
|
*/
|
|
void
|
|
dlg_del_window(WINDOW *win)
|
|
{
|
|
DIALOG_WINDOWS *p, *q, *r;
|
|
|
|
/*
|
|
* If --keep-window was set, do not delete/repaint the windows.
|
|
*/
|
|
if (dialog_vars.keep_window)
|
|
return;
|
|
|
|
/* Leave the main window untouched if there are no background windows.
|
|
* We do this so the current window will not be cleared on exit, allowing
|
|
* things like the infobox demo to run without flicker.
|
|
*/
|
|
if (dialog_state.getc_callbacks != 0) {
|
|
touchwin(stdscr);
|
|
wnoutrefresh(stdscr);
|
|
}
|
|
|
|
for (p = dialog_state.all_windows, q = r = 0; p != 0; r = p, p = p->next) {
|
|
if (p->normal == win) {
|
|
q = p; /* found a match - should be only one */
|
|
if (r == 0) {
|
|
dialog_state.all_windows = p->next;
|
|
} else {
|
|
r->next = p->next;
|
|
}
|
|
} else {
|
|
if (p->shadow != 0) {
|
|
touchwin(p->shadow);
|
|
wnoutrefresh(p->shadow);
|
|
}
|
|
touchwin(p->normal);
|
|
wnoutrefresh(p->normal);
|
|
}
|
|
}
|
|
|
|
if (q) {
|
|
if (dialog_state.all_windows != 0)
|
|
erase_childs_shadow(q);
|
|
del_subwindows(q->normal);
|
|
dlg_unregister_window(q->normal);
|
|
delwin(q->normal);
|
|
free(q);
|
|
}
|
|
doupdate();
|
|
}
|
|
|
|
/*
|
|
* Create a window, optionally with a shadow.
|
|
*/
|
|
WINDOW *
|
|
dlg_new_window(int height, int width, int y, int x)
|
|
{
|
|
return dlg_new_modal_window(stdscr, height, width, y, x);
|
|
}
|
|
|
|
/*
|
|
* "Modal" windows differ from normal ones by having a shadow in a window
|
|
* separate from the standard screen.
|
|
*/
|
|
WINDOW *
|
|
dlg_new_modal_window(WINDOW *parent, int height, int width, int y, int x)
|
|
{
|
|
WINDOW *win;
|
|
DIALOG_WINDOWS *p = dlg_calloc(DIALOG_WINDOWS, 1);
|
|
|
|
(void) parent;
|
|
if (p == 0
|
|
|| (win = newwin(height, width, y, x)) == 0) {
|
|
dlg_exiterr("Can't make new window at (%d,%d), size (%d,%d).\n",
|
|
y, x, height, width);
|
|
}
|
|
p->next = dialog_state.all_windows;
|
|
p->normal = win;
|
|
dialog_state.all_windows = p;
|
|
#ifdef HAVE_COLOR
|
|
if (dialog_state.use_shadow) {
|
|
p->shadow = parent;
|
|
draw_childs_shadow(p);
|
|
}
|
|
#endif
|
|
|
|
(void) keypad(win, TRUE);
|
|
return win;
|
|
}
|
|
|
|
/*
|
|
* Move/Resize a window, optionally with a shadow.
|
|
*/
|
|
#ifdef KEY_RESIZE
|
|
void
|
|
dlg_move_window(WINDOW *win, int height, int width, int y, int x)
|
|
{
|
|
DIALOG_WINDOWS *p;
|
|
|
|
if (win != 0) {
|
|
dlg_ctl_size(height, width);
|
|
|
|
if ((p = find_window(win)) != 0) {
|
|
(void) wresize(win, height, width);
|
|
(void) mvwin(win, y, x);
|
|
#ifdef HAVE_COLOR
|
|
if (p->shadow != 0) {
|
|
if (dialog_state.use_shadow) {
|
|
(void) mvwin(p->shadow, y + SHADOW_ROWS, x + SHADOW_COLS);
|
|
} else {
|
|
p->shadow = 0;
|
|
}
|
|
}
|
|
#endif
|
|
(void) refresh();
|
|
|
|
#ifdef HAVE_COLOR
|
|
draw_childs_shadow(p);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
#endif /* KEY_RESIZE */
|
|
|
|
WINDOW *
|
|
dlg_sub_window(WINDOW *parent, int height, int width, int y, int x)
|
|
{
|
|
WINDOW *win;
|
|
|
|
if ((win = subwin(parent, height, width, y, x)) == 0) {
|
|
dlg_exiterr("Can't make sub-window at (%d,%d), size (%d,%d).\n",
|
|
y, x, height, width);
|
|
}
|
|
|
|
add_subwindow(parent, win);
|
|
(void) keypad(win, TRUE);
|
|
return win;
|
|
}
|
|
|
|
/* obsolete */
|
|
int
|
|
dlg_default_item(char **items, int llen)
|
|
{
|
|
int result = 0;
|
|
|
|
if (dialog_vars.default_item != 0) {
|
|
int count = 0;
|
|
while (*items != 0) {
|
|
if (!strcmp(dialog_vars.default_item, *items)) {
|
|
result = count;
|
|
break;
|
|
}
|
|
items += llen;
|
|
count++;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int
|
|
dlg_default_listitem(DIALOG_LISTITEM * items)
|
|
{
|
|
int result = 0;
|
|
|
|
if (dialog_vars.default_item != 0) {
|
|
int count = 0;
|
|
while (items->name != 0) {
|
|
if (!strcmp(dialog_vars.default_item, items->name)) {
|
|
result = count;
|
|
break;
|
|
}
|
|
++items;
|
|
count++;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Draw the string for item_help
|
|
*/
|
|
void
|
|
dlg_item_help(const char *txt)
|
|
{
|
|
if (USE_ITEM_HELP(txt)) {
|
|
chtype attr = A_NORMAL;
|
|
int y, x;
|
|
|
|
(void) wattrset(stdscr, itemhelp_attr);
|
|
(void) wmove(stdscr, LINES - 1, 0);
|
|
(void) wclrtoeol(stdscr);
|
|
(void) addch(' ');
|
|
dlg_print_text(stdscr, txt, COLS - 1, &attr);
|
|
if (itemhelp_attr & A_COLOR) {
|
|
/* fill the remainder of the line with the window's attributes */
|
|
getyx(stdscr, y, x);
|
|
(void) y;
|
|
while (x < COLS) {
|
|
(void) addch(' ');
|
|
++x;
|
|
}
|
|
}
|
|
(void) wnoutrefresh(stdscr);
|
|
}
|
|
}
|
|
|
|
#ifndef HAVE_STRCASECMP
|
|
int
|
|
dlg_strcmp(const char *a, const char *b)
|
|
{
|
|
int ac, bc, cmp;
|
|
|
|
for (;;) {
|
|
ac = UCH(*a++);
|
|
bc = UCH(*b++);
|
|
if (isalpha(ac) && islower(ac))
|
|
ac = _toupper(ac);
|
|
if (isalpha(bc) && islower(bc))
|
|
bc = _toupper(bc);
|
|
cmp = ac - bc;
|
|
if (ac == 0 || bc == 0 || cmp != 0)
|
|
break;
|
|
}
|
|
return cmp;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Returns true if 'dst' points to a blank which follows another blank which
|
|
* is not a leading blank on a line.
|
|
*/
|
|
static bool
|
|
trim_blank(char *base, char *dst)
|
|
{
|
|
int count = 0;
|
|
|
|
while (dst-- != base) {
|
|
if (*dst == '\n') {
|
|
return FALSE;
|
|
} else if (*dst != ' ') {
|
|
return (count > 1);
|
|
} else {
|
|
count++;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* Change embedded "\n" substrings to '\n' characters and tabs to single
|
|
* spaces. If there are no "\n"s, it will strip all extra spaces, for
|
|
* justification. If it has "\n"'s, it will preserve extra spaces. If cr_wrap
|
|
* is set, it will preserve '\n's.
|
|
*/
|
|
void
|
|
dlg_trim_string(char *s)
|
|
{
|
|
char *base = s;
|
|
char *p1;
|
|
char *p = s;
|
|
int has_newlines = !dialog_vars.no_nl_expand && (strstr(s, "\\n") != 0);
|
|
|
|
while (*p != '\0') {
|
|
if (*p == TAB && !dialog_vars.nocollapse)
|
|
*p = ' ';
|
|
|
|
if (has_newlines) { /* If prompt contains "\n" strings */
|
|
if (*p == '\\' && *(p + 1) == 'n') {
|
|
*s++ = '\n';
|
|
p += 2;
|
|
p1 = p;
|
|
/*
|
|
* Handle end of lines intelligently. If '\n' follows "\n"
|
|
* then ignore the '\n'. This eliminates the need to escape
|
|
* the '\n' character (no need to use "\n\").
|
|
*/
|
|
while (*p1 == ' ')
|
|
p1++;
|
|
if (*p1 == '\n')
|
|
p = p1 + 1;
|
|
} else if (*p == '\n') {
|
|
if (dialog_vars.cr_wrap)
|
|
*s++ = *p++;
|
|
else {
|
|
/* Replace the '\n' with a space if cr_wrap is not set */
|
|
if (!trim_blank(base, s))
|
|
*s++ = ' ';
|
|
p++;
|
|
}
|
|
} else /* If *p != '\n' */
|
|
*s++ = *p++;
|
|
} else if (dialog_vars.trim_whitespace) {
|
|
if (*p == ' ') {
|
|
if (*(s - 1) != ' ') {
|
|
*s++ = ' ';
|
|
p++;
|
|
} else
|
|
p++;
|
|
} else if (*p == '\n') {
|
|
if (dialog_vars.cr_wrap)
|
|
*s++ = *p++;
|
|
else if (*(s - 1) != ' ') {
|
|
/* Strip '\n's if cr_wrap is not set. */
|
|
*s++ = ' ';
|
|
p++;
|
|
} else
|
|
p++;
|
|
} else
|
|
*s++ = *p++;
|
|
} else { /* If there are no "\n" strings */
|
|
if (*p == ' ' && !dialog_vars.nocollapse) {
|
|
if (!trim_blank(base, s))
|
|
*s++ = *p;
|
|
p++;
|
|
} else
|
|
*s++ = *p++;
|
|
}
|
|
}
|
|
|
|
*s = '\0';
|
|
}
|
|
|
|
void
|
|
dlg_set_focus(WINDOW *parent, WINDOW *win)
|
|
{
|
|
if (win != 0) {
|
|
(void) wmove(parent,
|
|
getpary(win) + getcury(win),
|
|
getparx(win) + getcurx(win));
|
|
(void) wnoutrefresh(win);
|
|
(void) doupdate();
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Returns the nominal maximum buffer size.
|
|
*/
|
|
int
|
|
dlg_max_input(int max_len)
|
|
{
|
|
if (dialog_vars.max_input != 0 && dialog_vars.max_input < MAX_LEN)
|
|
max_len = dialog_vars.max_input;
|
|
|
|
return max_len;
|
|
}
|
|
|
|
/*
|
|
* Free storage used for the result buffer.
|
|
*/
|
|
void
|
|
dlg_clr_result(void)
|
|
{
|
|
if (dialog_vars.input_length) {
|
|
dialog_vars.input_length = 0;
|
|
if (dialog_vars.input_result)
|
|
free(dialog_vars.input_result);
|
|
}
|
|
dialog_vars.input_result = 0;
|
|
}
|
|
|
|
/*
|
|
* Setup a fixed-buffer for the result.
|
|
*/
|
|
char *
|
|
dlg_set_result(const char *string)
|
|
{
|
|
unsigned need = string ? (unsigned) strlen(string) + 1 : 0;
|
|
|
|
/* inputstr.c needs a fixed buffer */
|
|
if (need < MAX_LEN)
|
|
need = MAX_LEN;
|
|
|
|
/*
|
|
* If the buffer is not big enough, allocate a new one.
|
|
*/
|
|
if (dialog_vars.input_length != 0
|
|
|| dialog_vars.input_result == 0
|
|
|| need > MAX_LEN) {
|
|
|
|
dlg_clr_result();
|
|
|
|
dialog_vars.input_length = need;
|
|
dialog_vars.input_result = dlg_malloc(char, need);
|
|
assert_ptr(dialog_vars.input_result, "dlg_set_result");
|
|
}
|
|
|
|
strcpy(dialog_vars.input_result, string ? string : "");
|
|
|
|
return dialog_vars.input_result;
|
|
}
|
|
|
|
/*
|
|
* Accumulate results in dynamically allocated buffer.
|
|
* If input_length is zero, it is a MAX_LEN buffer belonging to the caller.
|
|
*/
|
|
void
|
|
dlg_add_result(const char *string)
|
|
{
|
|
unsigned have = (dialog_vars.input_result
|
|
? (unsigned) strlen(dialog_vars.input_result)
|
|
: 0);
|
|
unsigned want = (unsigned) strlen(string) + 1 + have;
|
|
|
|
if ((want >= MAX_LEN)
|
|
|| (dialog_vars.input_length != 0)
|
|
|| (dialog_vars.input_result == 0)) {
|
|
|
|
if (dialog_vars.input_length == 0
|
|
|| dialog_vars.input_result == 0) {
|
|
|
|
char *save_result = dialog_vars.input_result;
|
|
|
|
dialog_vars.input_length = want * 2;
|
|
dialog_vars.input_result = dlg_malloc(char, dialog_vars.input_length);
|
|
assert_ptr(dialog_vars.input_result, "dlg_add_result malloc");
|
|
dialog_vars.input_result[0] = '\0';
|
|
if (save_result != 0)
|
|
strcpy(dialog_vars.input_result, save_result);
|
|
} else if (want >= dialog_vars.input_length) {
|
|
dialog_vars.input_length = want * 2;
|
|
dialog_vars.input_result = dlg_realloc(char,
|
|
dialog_vars.input_length,
|
|
dialog_vars.input_result);
|
|
assert_ptr(dialog_vars.input_result, "dlg_add_result realloc");
|
|
}
|
|
}
|
|
strcat(dialog_vars.input_result, string);
|
|
}
|
|
|
|
/*
|
|
* These are characters that (aside from the quote-delimiter) will have to
|
|
* be escaped in a single- or double-quoted string.
|
|
*/
|
|
#define FIX_SINGLE "\n\\"
|
|
#define FIX_DOUBLE FIX_SINGLE "[]{}?*;`~#$^&()|<>"
|
|
|
|
/*
|
|
* Returns the quote-delimiter.
|
|
*/
|
|
static const char *
|
|
quote_delimiter(void)
|
|
{
|
|
return dialog_vars.single_quoted ? "'" : "\"";
|
|
}
|
|
|
|
/*
|
|
* Returns true if we should quote the given string.
|
|
*/
|
|
static bool
|
|
must_quote(char *string)
|
|
{
|
|
bool code = FALSE;
|
|
|
|
if (*string != '\0') {
|
|
size_t len = strlen(string);
|
|
if (strcspn(string, quote_delimiter()) != len)
|
|
code = TRUE;
|
|
else if (strcspn(string, "\n\t ") != len)
|
|
code = TRUE;
|
|
else
|
|
code = (strcspn(string, FIX_DOUBLE) != len);
|
|
} else {
|
|
code = TRUE;
|
|
}
|
|
|
|
return code;
|
|
}
|
|
|
|
/*
|
|
* Add a quoted string to the result buffer.
|
|
*/
|
|
void
|
|
dlg_add_quoted(char *string)
|
|
{
|
|
char temp[2];
|
|
const char *my_quote = quote_delimiter();
|
|
const char *must_fix = (dialog_vars.single_quoted
|
|
? FIX_SINGLE
|
|
: FIX_DOUBLE);
|
|
|
|
if (must_quote(string)) {
|
|
temp[1] = '\0';
|
|
dlg_add_result(my_quote);
|
|
while (*string != '\0') {
|
|
temp[0] = *string++;
|
|
if (strchr(my_quote, *temp) || strchr(must_fix, *temp))
|
|
dlg_add_result("\\");
|
|
dlg_add_result(temp);
|
|
}
|
|
dlg_add_result(my_quote);
|
|
} else {
|
|
dlg_add_result(string);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* When adding a result, make that depend on whether "--quoted" is used.
|
|
*/
|
|
void
|
|
dlg_add_string(char *string)
|
|
{
|
|
if (dialog_vars.quoted) {
|
|
dlg_add_quoted(string);
|
|
} else {
|
|
dlg_add_result(string);
|
|
}
|
|
}
|
|
|
|
bool
|
|
dlg_need_separator(void)
|
|
{
|
|
bool result = FALSE;
|
|
|
|
if (dialog_vars.output_separator) {
|
|
result = TRUE;
|
|
} else if (dialog_vars.input_result && *(dialog_vars.input_result)) {
|
|
result = TRUE;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void
|
|
dlg_add_separator(void)
|
|
{
|
|
const char *separator = (dialog_vars.separate_output) ? "\n" : " ";
|
|
|
|
if (dialog_vars.output_separator)
|
|
separator = dialog_vars.output_separator;
|
|
|
|
dlg_add_result(separator);
|
|
}
|
|
|
|
#define HELP_PREFIX "HELP "
|
|
|
|
void
|
|
dlg_add_help_listitem(int *result, char **tag, DIALOG_LISTITEM * item)
|
|
{
|
|
dlg_add_result(HELP_PREFIX);
|
|
if (USE_ITEM_HELP(item->help)) {
|
|
*tag = dialog_vars.help_tags ? item->name : item->help;
|
|
*result = DLG_EXIT_ITEM_HELP;
|
|
} else {
|
|
*tag = item->name;
|
|
}
|
|
}
|
|
|
|
void
|
|
dlg_add_help_formitem(int *result, char **tag, DIALOG_FORMITEM * item)
|
|
{
|
|
dlg_add_result(HELP_PREFIX);
|
|
if (USE_ITEM_HELP(item->help)) {
|
|
*tag = dialog_vars.help_tags ? item->name : item->help;
|
|
*result = DLG_EXIT_ITEM_HELP;
|
|
} else {
|
|
*tag = item->name;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Some widgets support only one value of a given variable - save/restore the
|
|
* global dialog_vars so we can override it consistently.
|
|
*/
|
|
void
|
|
dlg_save_vars(DIALOG_VARS * vars)
|
|
{
|
|
*vars = dialog_vars;
|
|
}
|
|
|
|
/*
|
|
* Most of the data in DIALOG_VARS is normally set by command-line options.
|
|
* The input_result member is an exception; it is normally set by the dialog
|
|
* library to return result values.
|
|
*/
|
|
void
|
|
dlg_restore_vars(DIALOG_VARS * vars)
|
|
{
|
|
char *save_result = dialog_vars.input_result;
|
|
unsigned save_length = dialog_vars.input_length;
|
|
|
|
dialog_vars = *vars;
|
|
dialog_vars.input_result = save_result;
|
|
dialog_vars.input_length = save_length;
|
|
}
|
|
|
|
/*
|
|
* Called each time a widget is invoked which may do output, increment a count.
|
|
*/
|
|
void
|
|
dlg_does_output(void)
|
|
{
|
|
dialog_state.output_count += 1;
|
|
}
|
|
|
|
/*
|
|
* Compatibility for different versions of curses.
|
|
*/
|
|
#if !(defined(HAVE_GETBEGX) && defined(HAVE_GETBEGY))
|
|
int
|
|
dlg_getbegx(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getbegyx(win, y, x);
|
|
return x;
|
|
}
|
|
int
|
|
dlg_getbegy(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getbegyx(win, y, x);
|
|
return y;
|
|
}
|
|
#endif
|
|
|
|
#if !(defined(HAVE_GETCURX) && defined(HAVE_GETCURY))
|
|
int
|
|
dlg_getcurx(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getyx(win, y, x);
|
|
return x;
|
|
}
|
|
int
|
|
dlg_getcury(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getyx(win, y, x);
|
|
return y;
|
|
}
|
|
#endif
|
|
|
|
#if !(defined(HAVE_GETMAXX) && defined(HAVE_GETMAXY))
|
|
int
|
|
dlg_getmaxx(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getmaxyx(win, y, x);
|
|
return x;
|
|
}
|
|
int
|
|
dlg_getmaxy(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getmaxyx(win, y, x);
|
|
return y;
|
|
}
|
|
#endif
|
|
|
|
#if !(defined(HAVE_GETPARX) && defined(HAVE_GETPARY))
|
|
int
|
|
dlg_getparx(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getparyx(win, y, x);
|
|
return x;
|
|
}
|
|
int
|
|
dlg_getpary(WINDOW *win)
|
|
{
|
|
int y, x;
|
|
getparyx(win, y, x);
|
|
return y;
|
|
}
|
|
#endif
|
|
|
|
#ifdef NEED_WGETPARENT
|
|
WINDOW *
|
|
dlg_wgetparent(WINDOW *win)
|
|
{
|
|
#undef wgetparent
|
|
WINDOW *result = 0;
|
|
DIALOG_WINDOWS *p;
|
|
|
|
for (p = dialog_state.all_subwindows; p != 0; p = p->next) {
|
|
if (p->shadow == win) {
|
|
result = p->normal;
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
#endif
|