xref: /nrf52832-nimble/rt-thread/tools/kconfig-frontends/libs/lxdialog/util.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  *  util.c
3*10465441SEvalZero  *
4*10465441SEvalZero  *  ORIGINAL AUTHOR: Savio Lam ([email protected])
5*10465441SEvalZero  *  MODIFIED FOR LINUX KERNEL CONFIG BY: William Roadcap ([email protected])
6*10465441SEvalZero  *
7*10465441SEvalZero  *  This program is free software; you can redistribute it and/or
8*10465441SEvalZero  *  modify it under the terms of the GNU General Public License
9*10465441SEvalZero  *  as published by the Free Software Foundation; either version 2
10*10465441SEvalZero  *  of the License, or (at your option) any later version.
11*10465441SEvalZero  *
12*10465441SEvalZero  *  This program is distributed in the hope that it will be useful,
13*10465441SEvalZero  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
14*10465441SEvalZero  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15*10465441SEvalZero  *  GNU General Public License for more details.
16*10465441SEvalZero  *
17*10465441SEvalZero  *  You should have received a copy of the GNU General Public License
18*10465441SEvalZero  *  along with this program; if not, write to the Free Software
19*10465441SEvalZero  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20*10465441SEvalZero  */
21*10465441SEvalZero 
22*10465441SEvalZero #include <stdarg.h>
23*10465441SEvalZero 
24*10465441SEvalZero #include "dialog.h"
25*10465441SEvalZero 
26*10465441SEvalZero /* Needed in signal handler in mconf.c */
27*10465441SEvalZero int saved_x, saved_y;
28*10465441SEvalZero 
29*10465441SEvalZero struct dialog_info dlg;
30*10465441SEvalZero 
set_mono_theme(void)31*10465441SEvalZero static void set_mono_theme(void)
32*10465441SEvalZero {
33*10465441SEvalZero 	dlg.screen.atr = A_NORMAL;
34*10465441SEvalZero 	dlg.shadow.atr = A_NORMAL;
35*10465441SEvalZero 	dlg.dialog.atr = A_NORMAL;
36*10465441SEvalZero 	dlg.title.atr = A_BOLD;
37*10465441SEvalZero 	dlg.border.atr = A_NORMAL;
38*10465441SEvalZero 	dlg.button_active.atr = A_REVERSE;
39*10465441SEvalZero 	dlg.button_inactive.atr = A_DIM;
40*10465441SEvalZero 	dlg.button_key_active.atr = A_REVERSE;
41*10465441SEvalZero 	dlg.button_key_inactive.atr = A_BOLD;
42*10465441SEvalZero 	dlg.button_label_active.atr = A_REVERSE;
43*10465441SEvalZero 	dlg.button_label_inactive.atr = A_NORMAL;
44*10465441SEvalZero 	dlg.inputbox.atr = A_NORMAL;
45*10465441SEvalZero 	dlg.inputbox_border.atr = A_NORMAL;
46*10465441SEvalZero 	dlg.searchbox.atr = A_NORMAL;
47*10465441SEvalZero 	dlg.searchbox_title.atr = A_BOLD;
48*10465441SEvalZero 	dlg.searchbox_border.atr = A_NORMAL;
49*10465441SEvalZero 	dlg.position_indicator.atr = A_BOLD;
50*10465441SEvalZero 	dlg.menubox.atr = A_NORMAL;
51*10465441SEvalZero 	dlg.menubox_border.atr = A_NORMAL;
52*10465441SEvalZero 	dlg.item.atr = A_NORMAL;
53*10465441SEvalZero 	dlg.item_selected.atr = A_REVERSE;
54*10465441SEvalZero 	dlg.tag.atr = A_BOLD;
55*10465441SEvalZero 	dlg.tag_selected.atr = A_REVERSE;
56*10465441SEvalZero 	dlg.tag_key.atr = A_BOLD;
57*10465441SEvalZero 	dlg.tag_key_selected.atr = A_REVERSE;
58*10465441SEvalZero 	dlg.check.atr = A_BOLD;
59*10465441SEvalZero 	dlg.check_selected.atr = A_REVERSE;
60*10465441SEvalZero 	dlg.uarrow.atr = A_BOLD;
61*10465441SEvalZero 	dlg.darrow.atr = A_BOLD;
62*10465441SEvalZero }
63*10465441SEvalZero 
64*10465441SEvalZero #define DLG_COLOR(dialog, f, b, h) \
65*10465441SEvalZero do {                               \
66*10465441SEvalZero 	dlg.dialog.fg = (f);       \
67*10465441SEvalZero 	dlg.dialog.bg = (b);       \
68*10465441SEvalZero 	dlg.dialog.hl = (h);       \
69*10465441SEvalZero } while (0)
70*10465441SEvalZero 
set_classic_theme(void)71*10465441SEvalZero static void set_classic_theme(void)
72*10465441SEvalZero {
73*10465441SEvalZero 	DLG_COLOR(screen,                COLOR_CYAN,   COLOR_BLUE,   true);
74*10465441SEvalZero 	DLG_COLOR(shadow,                COLOR_BLACK,  COLOR_BLACK,  true);
75*10465441SEvalZero 	DLG_COLOR(dialog,                COLOR_BLACK,  COLOR_WHITE,  false);
76*10465441SEvalZero 	DLG_COLOR(title,                 COLOR_YELLOW, COLOR_WHITE,  true);
77*10465441SEvalZero 	DLG_COLOR(border,                COLOR_WHITE,  COLOR_WHITE,  true);
78*10465441SEvalZero 	DLG_COLOR(button_active,         COLOR_WHITE,  COLOR_BLUE,   true);
79*10465441SEvalZero 	DLG_COLOR(button_inactive,       COLOR_BLACK,  COLOR_WHITE,  false);
80*10465441SEvalZero 	DLG_COLOR(button_key_active,     COLOR_WHITE,  COLOR_BLUE,   true);
81*10465441SEvalZero 	DLG_COLOR(button_key_inactive,   COLOR_RED,    COLOR_WHITE,  false);
82*10465441SEvalZero 	DLG_COLOR(button_label_active,   COLOR_YELLOW, COLOR_BLUE,   true);
83*10465441SEvalZero 	DLG_COLOR(button_label_inactive, COLOR_BLACK,  COLOR_WHITE,  true);
84*10465441SEvalZero 	DLG_COLOR(inputbox,              COLOR_BLACK,  COLOR_WHITE,  false);
85*10465441SEvalZero 	DLG_COLOR(inputbox_border,       COLOR_BLACK,  COLOR_WHITE,  false);
86*10465441SEvalZero 	DLG_COLOR(searchbox,             COLOR_BLACK,  COLOR_WHITE,  false);
87*10465441SEvalZero 	DLG_COLOR(searchbox_title,       COLOR_YELLOW, COLOR_WHITE,  true);
88*10465441SEvalZero 	DLG_COLOR(searchbox_border,      COLOR_WHITE,  COLOR_WHITE,  true);
89*10465441SEvalZero 	DLG_COLOR(position_indicator,    COLOR_YELLOW, COLOR_WHITE,  true);
90*10465441SEvalZero 	DLG_COLOR(menubox,               COLOR_BLACK,  COLOR_WHITE,  false);
91*10465441SEvalZero 	DLG_COLOR(menubox_border,        COLOR_WHITE,  COLOR_WHITE,  true);
92*10465441SEvalZero 	DLG_COLOR(item,                  COLOR_BLACK,  COLOR_WHITE,  false);
93*10465441SEvalZero 	DLG_COLOR(item_selected,         COLOR_WHITE,  COLOR_BLUE,   true);
94*10465441SEvalZero 	DLG_COLOR(tag,                   COLOR_YELLOW, COLOR_WHITE,  true);
95*10465441SEvalZero 	DLG_COLOR(tag_selected,          COLOR_YELLOW, COLOR_BLUE,   true);
96*10465441SEvalZero 	DLG_COLOR(tag_key,               COLOR_YELLOW, COLOR_WHITE,  true);
97*10465441SEvalZero 	DLG_COLOR(tag_key_selected,      COLOR_YELLOW, COLOR_BLUE,   true);
98*10465441SEvalZero 	DLG_COLOR(check,                 COLOR_BLACK,  COLOR_WHITE,  false);
99*10465441SEvalZero 	DLG_COLOR(check_selected,        COLOR_WHITE,  COLOR_BLUE,   true);
100*10465441SEvalZero 	DLG_COLOR(uarrow,                COLOR_GREEN,  COLOR_WHITE,  true);
101*10465441SEvalZero 	DLG_COLOR(darrow,                COLOR_GREEN,  COLOR_WHITE,  true);
102*10465441SEvalZero }
103*10465441SEvalZero 
set_blackbg_theme(void)104*10465441SEvalZero static void set_blackbg_theme(void)
105*10465441SEvalZero {
106*10465441SEvalZero 	DLG_COLOR(screen, COLOR_RED,   COLOR_BLACK, true);
107*10465441SEvalZero 	DLG_COLOR(shadow, COLOR_BLACK, COLOR_BLACK, false);
108*10465441SEvalZero 	DLG_COLOR(dialog, COLOR_WHITE, COLOR_BLACK, false);
109*10465441SEvalZero 	DLG_COLOR(title,  COLOR_RED,   COLOR_BLACK, false);
110*10465441SEvalZero 	DLG_COLOR(border, COLOR_BLACK, COLOR_BLACK, true);
111*10465441SEvalZero 
112*10465441SEvalZero 	DLG_COLOR(button_active,         COLOR_YELLOW, COLOR_RED,   false);
113*10465441SEvalZero 	DLG_COLOR(button_inactive,       COLOR_YELLOW, COLOR_BLACK, false);
114*10465441SEvalZero 	DLG_COLOR(button_key_active,     COLOR_YELLOW, COLOR_RED,   true);
115*10465441SEvalZero 	DLG_COLOR(button_key_inactive,   COLOR_RED,    COLOR_BLACK, false);
116*10465441SEvalZero 	DLG_COLOR(button_label_active,   COLOR_WHITE,  COLOR_RED,   false);
117*10465441SEvalZero 	DLG_COLOR(button_label_inactive, COLOR_BLACK,  COLOR_BLACK, true);
118*10465441SEvalZero 
119*10465441SEvalZero 	DLG_COLOR(inputbox,         COLOR_YELLOW, COLOR_BLACK, false);
120*10465441SEvalZero 	DLG_COLOR(inputbox_border,  COLOR_YELLOW, COLOR_BLACK, false);
121*10465441SEvalZero 
122*10465441SEvalZero 	DLG_COLOR(searchbox,        COLOR_YELLOW, COLOR_BLACK, false);
123*10465441SEvalZero 	DLG_COLOR(searchbox_title,  COLOR_YELLOW, COLOR_BLACK, true);
124*10465441SEvalZero 	DLG_COLOR(searchbox_border, COLOR_BLACK,  COLOR_BLACK, true);
125*10465441SEvalZero 
126*10465441SEvalZero 	DLG_COLOR(position_indicator, COLOR_RED, COLOR_BLACK,  false);
127*10465441SEvalZero 
128*10465441SEvalZero 	DLG_COLOR(menubox,          COLOR_YELLOW, COLOR_BLACK, false);
129*10465441SEvalZero 	DLG_COLOR(menubox_border,   COLOR_BLACK,  COLOR_BLACK, true);
130*10465441SEvalZero 
131*10465441SEvalZero 	DLG_COLOR(item,             COLOR_WHITE, COLOR_BLACK, false);
132*10465441SEvalZero 	DLG_COLOR(item_selected,    COLOR_WHITE, COLOR_RED,   false);
133*10465441SEvalZero 
134*10465441SEvalZero 	DLG_COLOR(tag,              COLOR_RED,    COLOR_BLACK, false);
135*10465441SEvalZero 	DLG_COLOR(tag_selected,     COLOR_YELLOW, COLOR_RED,   true);
136*10465441SEvalZero 	DLG_COLOR(tag_key,          COLOR_RED,    COLOR_BLACK, false);
137*10465441SEvalZero 	DLG_COLOR(tag_key_selected, COLOR_YELLOW, COLOR_RED,   true);
138*10465441SEvalZero 
139*10465441SEvalZero 	DLG_COLOR(check,            COLOR_YELLOW, COLOR_BLACK, false);
140*10465441SEvalZero 	DLG_COLOR(check_selected,   COLOR_YELLOW, COLOR_RED,   true);
141*10465441SEvalZero 
142*10465441SEvalZero 	DLG_COLOR(uarrow, COLOR_RED, COLOR_BLACK, false);
143*10465441SEvalZero 	DLG_COLOR(darrow, COLOR_RED, COLOR_BLACK, false);
144*10465441SEvalZero }
145*10465441SEvalZero 
set_bluetitle_theme(void)146*10465441SEvalZero static void set_bluetitle_theme(void)
147*10465441SEvalZero {
148*10465441SEvalZero 	set_classic_theme();
149*10465441SEvalZero 	DLG_COLOR(title,               COLOR_BLUE,   COLOR_WHITE, true);
150*10465441SEvalZero 	DLG_COLOR(button_key_active,   COLOR_YELLOW, COLOR_BLUE,  true);
151*10465441SEvalZero 	DLG_COLOR(button_label_active, COLOR_WHITE,  COLOR_BLUE,  true);
152*10465441SEvalZero 	DLG_COLOR(searchbox_title,     COLOR_BLUE,   COLOR_WHITE, true);
153*10465441SEvalZero 	DLG_COLOR(position_indicator,  COLOR_BLUE,   COLOR_WHITE, true);
154*10465441SEvalZero 	DLG_COLOR(tag,                 COLOR_BLUE,   COLOR_WHITE, true);
155*10465441SEvalZero 	DLG_COLOR(tag_key,             COLOR_BLUE,   COLOR_WHITE, true);
156*10465441SEvalZero 
157*10465441SEvalZero }
158*10465441SEvalZero 
159*10465441SEvalZero /*
160*10465441SEvalZero  * Select color theme
161*10465441SEvalZero  */
set_theme(const char * theme)162*10465441SEvalZero static int set_theme(const char *theme)
163*10465441SEvalZero {
164*10465441SEvalZero 	int use_color = 1;
165*10465441SEvalZero 	if (!theme)
166*10465441SEvalZero 		set_bluetitle_theme();
167*10465441SEvalZero 	else if (strcmp(theme, "classic") == 0)
168*10465441SEvalZero 		set_classic_theme();
169*10465441SEvalZero 	else if (strcmp(theme, "bluetitle") == 0)
170*10465441SEvalZero 		set_bluetitle_theme();
171*10465441SEvalZero 	else if (strcmp(theme, "blackbg") == 0)
172*10465441SEvalZero 		set_blackbg_theme();
173*10465441SEvalZero 	else if (strcmp(theme, "mono") == 0)
174*10465441SEvalZero 		use_color = 0;
175*10465441SEvalZero 
176*10465441SEvalZero 	return use_color;
177*10465441SEvalZero }
178*10465441SEvalZero 
init_one_color(struct dialog_color * color)179*10465441SEvalZero static void init_one_color(struct dialog_color *color)
180*10465441SEvalZero {
181*10465441SEvalZero 	static int pair = 0;
182*10465441SEvalZero 
183*10465441SEvalZero 	pair++;
184*10465441SEvalZero 	init_pair(pair, color->fg, color->bg);
185*10465441SEvalZero 	if (color->hl)
186*10465441SEvalZero 		color->atr = A_BOLD | COLOR_PAIR(pair);
187*10465441SEvalZero 	else
188*10465441SEvalZero 		color->atr = COLOR_PAIR(pair);
189*10465441SEvalZero }
190*10465441SEvalZero 
init_dialog_colors(void)191*10465441SEvalZero static void init_dialog_colors(void)
192*10465441SEvalZero {
193*10465441SEvalZero 	init_one_color(&dlg.screen);
194*10465441SEvalZero 	init_one_color(&dlg.shadow);
195*10465441SEvalZero 	init_one_color(&dlg.dialog);
196*10465441SEvalZero 	init_one_color(&dlg.title);
197*10465441SEvalZero 	init_one_color(&dlg.border);
198*10465441SEvalZero 	init_one_color(&dlg.button_active);
199*10465441SEvalZero 	init_one_color(&dlg.button_inactive);
200*10465441SEvalZero 	init_one_color(&dlg.button_key_active);
201*10465441SEvalZero 	init_one_color(&dlg.button_key_inactive);
202*10465441SEvalZero 	init_one_color(&dlg.button_label_active);
203*10465441SEvalZero 	init_one_color(&dlg.button_label_inactive);
204*10465441SEvalZero 	init_one_color(&dlg.inputbox);
205*10465441SEvalZero 	init_one_color(&dlg.inputbox_border);
206*10465441SEvalZero 	init_one_color(&dlg.searchbox);
207*10465441SEvalZero 	init_one_color(&dlg.searchbox_title);
208*10465441SEvalZero 	init_one_color(&dlg.searchbox_border);
209*10465441SEvalZero 	init_one_color(&dlg.position_indicator);
210*10465441SEvalZero 	init_one_color(&dlg.menubox);
211*10465441SEvalZero 	init_one_color(&dlg.menubox_border);
212*10465441SEvalZero 	init_one_color(&dlg.item);
213*10465441SEvalZero 	init_one_color(&dlg.item_selected);
214*10465441SEvalZero 	init_one_color(&dlg.tag);
215*10465441SEvalZero 	init_one_color(&dlg.tag_selected);
216*10465441SEvalZero 	init_one_color(&dlg.tag_key);
217*10465441SEvalZero 	init_one_color(&dlg.tag_key_selected);
218*10465441SEvalZero 	init_one_color(&dlg.check);
219*10465441SEvalZero 	init_one_color(&dlg.check_selected);
220*10465441SEvalZero 	init_one_color(&dlg.uarrow);
221*10465441SEvalZero 	init_one_color(&dlg.darrow);
222*10465441SEvalZero }
223*10465441SEvalZero 
224*10465441SEvalZero /*
225*10465441SEvalZero  * Setup for color display
226*10465441SEvalZero  */
color_setup(const char * theme)227*10465441SEvalZero static void color_setup(const char *theme)
228*10465441SEvalZero {
229*10465441SEvalZero 	int use_color;
230*10465441SEvalZero 
231*10465441SEvalZero 	use_color = set_theme(theme);
232*10465441SEvalZero 	if (use_color && has_colors()) {
233*10465441SEvalZero 		start_color();
234*10465441SEvalZero 		init_dialog_colors();
235*10465441SEvalZero 	} else
236*10465441SEvalZero 		set_mono_theme();
237*10465441SEvalZero }
238*10465441SEvalZero 
239*10465441SEvalZero /*
240*10465441SEvalZero  * Set window to attribute 'attr'
241*10465441SEvalZero  */
attr_clear(WINDOW * win,int height,int width,chtype attr)242*10465441SEvalZero void attr_clear(WINDOW * win, int height, int width, chtype attr)
243*10465441SEvalZero {
244*10465441SEvalZero 	int i, j;
245*10465441SEvalZero 
246*10465441SEvalZero 	wattrset(win, attr);
247*10465441SEvalZero 	for (i = 0; i < height; i++) {
248*10465441SEvalZero 		wmove(win, i, 0);
249*10465441SEvalZero 		for (j = 0; j < width; j++)
250*10465441SEvalZero 			waddch(win, ' ');
251*10465441SEvalZero 	}
252*10465441SEvalZero 	touchwin(win);
253*10465441SEvalZero }
254*10465441SEvalZero 
dialog_clear(void)255*10465441SEvalZero void dialog_clear(void)
256*10465441SEvalZero {
257*10465441SEvalZero 	int lines, columns;
258*10465441SEvalZero 
259*10465441SEvalZero 	lines = getmaxy(stdscr);
260*10465441SEvalZero 	columns = getmaxx(stdscr);
261*10465441SEvalZero 
262*10465441SEvalZero 	attr_clear(stdscr, lines, columns, dlg.screen.atr);
263*10465441SEvalZero 	/* Display background title if it exists ... - SLH */
264*10465441SEvalZero 	if (dlg.backtitle != NULL) {
265*10465441SEvalZero 		int i, len = 0, skip = 0;
266*10465441SEvalZero 		struct subtitle_list *pos;
267*10465441SEvalZero 
268*10465441SEvalZero 		wattrset(stdscr, dlg.screen.atr);
269*10465441SEvalZero 		mvwaddstr(stdscr, 0, 1, (char *)dlg.backtitle);
270*10465441SEvalZero 
271*10465441SEvalZero 		for (pos = dlg.subtitles; pos != NULL; pos = pos->next) {
272*10465441SEvalZero 			/* 3 is for the arrow and spaces */
273*10465441SEvalZero 			len += strlen(pos->text) + 3;
274*10465441SEvalZero 		}
275*10465441SEvalZero 
276*10465441SEvalZero 		wmove(stdscr, 1, 1);
277*10465441SEvalZero 		if (len > columns - 2) {
278*10465441SEvalZero 			const char *ellipsis = "[...] ";
279*10465441SEvalZero 			waddstr(stdscr, ellipsis);
280*10465441SEvalZero 			skip = len - (columns - 2 - strlen(ellipsis));
281*10465441SEvalZero 		}
282*10465441SEvalZero 
283*10465441SEvalZero 		for (pos = dlg.subtitles; pos != NULL; pos = pos->next) {
284*10465441SEvalZero 			if (skip == 0)
285*10465441SEvalZero 				waddch(stdscr, ACS_RARROW);
286*10465441SEvalZero 			else
287*10465441SEvalZero 				skip--;
288*10465441SEvalZero 
289*10465441SEvalZero 			if (skip == 0)
290*10465441SEvalZero 				waddch(stdscr, ' ');
291*10465441SEvalZero 			else
292*10465441SEvalZero 				skip--;
293*10465441SEvalZero 
294*10465441SEvalZero 			if (skip < strlen(pos->text)) {
295*10465441SEvalZero 				waddstr(stdscr, pos->text + skip);
296*10465441SEvalZero 				skip = 0;
297*10465441SEvalZero 			} else
298*10465441SEvalZero 				skip -= strlen(pos->text);
299*10465441SEvalZero 
300*10465441SEvalZero 			if (skip == 0)
301*10465441SEvalZero 				waddch(stdscr, ' ');
302*10465441SEvalZero 			else
303*10465441SEvalZero 				skip--;
304*10465441SEvalZero 		}
305*10465441SEvalZero 
306*10465441SEvalZero 		for (i = len + 1; i < columns - 1; i++)
307*10465441SEvalZero 			waddch(stdscr, ACS_HLINE);
308*10465441SEvalZero 	}
309*10465441SEvalZero 	wnoutrefresh(stdscr);
310*10465441SEvalZero }
311*10465441SEvalZero 
312*10465441SEvalZero /*
313*10465441SEvalZero  * Do some initialization for dialog
314*10465441SEvalZero  */
init_dialog(const char * backtitle)315*10465441SEvalZero int init_dialog(const char *backtitle)
316*10465441SEvalZero {
317*10465441SEvalZero 	int height, width;
318*10465441SEvalZero 
319*10465441SEvalZero 	initscr();		/* Init curses */
320*10465441SEvalZero 
321*10465441SEvalZero 	/* Get current cursor position for signal handler in mconf.c */
322*10465441SEvalZero 	getyx(stdscr, saved_y, saved_x);
323*10465441SEvalZero 
324*10465441SEvalZero 	getmaxyx(stdscr, height, width);
325*10465441SEvalZero 	if (height < WINDOW_HEIGTH_MIN || width < WINDOW_WIDTH_MIN) {
326*10465441SEvalZero 		endwin();
327*10465441SEvalZero 		return -ERRDISPLAYTOOSMALL;
328*10465441SEvalZero 	}
329*10465441SEvalZero 
330*10465441SEvalZero 	dlg.backtitle = backtitle;
331*10465441SEvalZero 	color_setup(getenv("MENUCONFIG_COLOR"));
332*10465441SEvalZero 
333*10465441SEvalZero 	keypad(stdscr, TRUE);
334*10465441SEvalZero 	cbreak();
335*10465441SEvalZero 	noecho();
336*10465441SEvalZero 	dialog_clear();
337*10465441SEvalZero 
338*10465441SEvalZero 	return 0;
339*10465441SEvalZero }
340*10465441SEvalZero 
set_dialog_backtitle(const char * backtitle)341*10465441SEvalZero void set_dialog_backtitle(const char *backtitle)
342*10465441SEvalZero {
343*10465441SEvalZero 	dlg.backtitle = backtitle;
344*10465441SEvalZero }
345*10465441SEvalZero 
set_dialog_subtitles(struct subtitle_list * subtitles)346*10465441SEvalZero void set_dialog_subtitles(struct subtitle_list *subtitles)
347*10465441SEvalZero {
348*10465441SEvalZero 	dlg.subtitles = subtitles;
349*10465441SEvalZero }
350*10465441SEvalZero 
351*10465441SEvalZero /*
352*10465441SEvalZero  * End using dialog functions.
353*10465441SEvalZero  */
end_dialog(int x,int y)354*10465441SEvalZero void end_dialog(int x, int y)
355*10465441SEvalZero {
356*10465441SEvalZero 	/* move cursor back to original position */
357*10465441SEvalZero 	move(y, x);
358*10465441SEvalZero 	refresh();
359*10465441SEvalZero 	endwin();
360*10465441SEvalZero }
361*10465441SEvalZero 
362*10465441SEvalZero /* Print the title of the dialog. Center the title and truncate
363*10465441SEvalZero  * tile if wider than dialog (- 2 chars).
364*10465441SEvalZero  **/
print_title(WINDOW * dialog,const char * title,int width)365*10465441SEvalZero void print_title(WINDOW *dialog, const char *title, int width)
366*10465441SEvalZero {
367*10465441SEvalZero 	if (title) {
368*10465441SEvalZero 		int tlen = MIN(width - 2, strlen(title));
369*10465441SEvalZero 		wattrset(dialog, dlg.title.atr);
370*10465441SEvalZero 		mvwaddch(dialog, 0, (width - tlen) / 2 - 1, ' ');
371*10465441SEvalZero 		mvwaddnstr(dialog, 0, (width - tlen)/2, title, tlen);
372*10465441SEvalZero 		waddch(dialog, ' ');
373*10465441SEvalZero 	}
374*10465441SEvalZero }
375*10465441SEvalZero 
376*10465441SEvalZero /*
377*10465441SEvalZero  * Print a string of text in a window, automatically wrap around to the
378*10465441SEvalZero  * next line if the string is too long to fit on one line. Newline
379*10465441SEvalZero  * characters '\n' are propperly processed.  We start on a new line
380*10465441SEvalZero  * if there is no room for at least 4 nonblanks following a double-space.
381*10465441SEvalZero  */
print_autowrap(WINDOW * win,const char * prompt,int width,int y,int x)382*10465441SEvalZero void print_autowrap(WINDOW * win, const char *prompt, int width, int y, int x)
383*10465441SEvalZero {
384*10465441SEvalZero 	int newl, cur_x, cur_y;
385*10465441SEvalZero 	int prompt_len, room, wlen;
386*10465441SEvalZero 	char tempstr[MAX_LEN + 1], *word, *sp, *sp2, *newline_separator = 0;
387*10465441SEvalZero 
388*10465441SEvalZero 	strcpy(tempstr, prompt);
389*10465441SEvalZero 
390*10465441SEvalZero 	prompt_len = strlen(tempstr);
391*10465441SEvalZero 
392*10465441SEvalZero 	if (prompt_len <= width - x * 2) {	/* If prompt is short */
393*10465441SEvalZero 		wmove(win, y, (width - prompt_len) / 2);
394*10465441SEvalZero 		waddstr(win, tempstr);
395*10465441SEvalZero 	} else {
396*10465441SEvalZero 		cur_x = x;
397*10465441SEvalZero 		cur_y = y;
398*10465441SEvalZero 		newl = 1;
399*10465441SEvalZero 		word = tempstr;
400*10465441SEvalZero 		while (word && *word) {
401*10465441SEvalZero 			sp = strpbrk(word, "\n ");
402*10465441SEvalZero 			if (sp && *sp == '\n')
403*10465441SEvalZero 				newline_separator = sp;
404*10465441SEvalZero 
405*10465441SEvalZero 			if (sp)
406*10465441SEvalZero 				*sp++ = 0;
407*10465441SEvalZero 
408*10465441SEvalZero 			/* Wrap to next line if either the word does not fit,
409*10465441SEvalZero 			   or it is the first word of a new sentence, and it is
410*10465441SEvalZero 			   short, and the next word does not fit. */
411*10465441SEvalZero 			room = width - cur_x;
412*10465441SEvalZero 			wlen = strlen(word);
413*10465441SEvalZero 			if (wlen > room ||
414*10465441SEvalZero 			    (newl && wlen < 4 && sp
415*10465441SEvalZero 			     && wlen + 1 + strlen(sp) > room
416*10465441SEvalZero 			     && (!(sp2 = strpbrk(sp, "\n "))
417*10465441SEvalZero 				 || wlen + 1 + (sp2 - sp) > room))) {
418*10465441SEvalZero 				cur_y++;
419*10465441SEvalZero 				cur_x = x;
420*10465441SEvalZero 			}
421*10465441SEvalZero 			wmove(win, cur_y, cur_x);
422*10465441SEvalZero 			waddstr(win, word);
423*10465441SEvalZero 			getyx(win, cur_y, cur_x);
424*10465441SEvalZero 
425*10465441SEvalZero 			/* Move to the next line if the word separator was a newline */
426*10465441SEvalZero 			if (newline_separator) {
427*10465441SEvalZero 				cur_y++;
428*10465441SEvalZero 				cur_x = x;
429*10465441SEvalZero 				newline_separator = 0;
430*10465441SEvalZero 			} else
431*10465441SEvalZero 				cur_x++;
432*10465441SEvalZero 
433*10465441SEvalZero 			if (sp && *sp == ' ') {
434*10465441SEvalZero 				cur_x++;	/* double space */
435*10465441SEvalZero 				while (*++sp == ' ') ;
436*10465441SEvalZero 				newl = 1;
437*10465441SEvalZero 			} else
438*10465441SEvalZero 				newl = 0;
439*10465441SEvalZero 			word = sp;
440*10465441SEvalZero 		}
441*10465441SEvalZero 	}
442*10465441SEvalZero }
443*10465441SEvalZero 
444*10465441SEvalZero /*
445*10465441SEvalZero  * Print a button
446*10465441SEvalZero  */
print_button(WINDOW * win,const char * label,int y,int x,int selected)447*10465441SEvalZero void print_button(WINDOW * win, const char *label, int y, int x, int selected)
448*10465441SEvalZero {
449*10465441SEvalZero 	int i, temp;
450*10465441SEvalZero 
451*10465441SEvalZero 	wmove(win, y, x);
452*10465441SEvalZero 	wattrset(win, selected ? dlg.button_active.atr
453*10465441SEvalZero 		 : dlg.button_inactive.atr);
454*10465441SEvalZero 	waddstr(win, "<");
455*10465441SEvalZero 	temp = strspn(label, " ");
456*10465441SEvalZero 	label += temp;
457*10465441SEvalZero 	wattrset(win, selected ? dlg.button_label_active.atr
458*10465441SEvalZero 		 : dlg.button_label_inactive.atr);
459*10465441SEvalZero 	for (i = 0; i < temp; i++)
460*10465441SEvalZero 		waddch(win, ' ');
461*10465441SEvalZero 	wattrset(win, selected ? dlg.button_key_active.atr
462*10465441SEvalZero 		 : dlg.button_key_inactive.atr);
463*10465441SEvalZero 	waddch(win, label[0]);
464*10465441SEvalZero 	wattrset(win, selected ? dlg.button_label_active.atr
465*10465441SEvalZero 		 : dlg.button_label_inactive.atr);
466*10465441SEvalZero 	waddstr(win, (char *)label + 1);
467*10465441SEvalZero 	wattrset(win, selected ? dlg.button_active.atr
468*10465441SEvalZero 		 : dlg.button_inactive.atr);
469*10465441SEvalZero 	waddstr(win, ">");
470*10465441SEvalZero 	wmove(win, y, x + temp + 1);
471*10465441SEvalZero }
472*10465441SEvalZero 
473*10465441SEvalZero /*
474*10465441SEvalZero  * Draw a rectangular box with line drawing characters
475*10465441SEvalZero  */
476*10465441SEvalZero void
draw_box(WINDOW * win,int y,int x,int height,int width,chtype box,chtype border)477*10465441SEvalZero draw_box(WINDOW * win, int y, int x, int height, int width,
478*10465441SEvalZero 	 chtype box, chtype border)
479*10465441SEvalZero {
480*10465441SEvalZero 	int i, j;
481*10465441SEvalZero 
482*10465441SEvalZero 	wattrset(win, 0);
483*10465441SEvalZero 	for (i = 0; i < height; i++) {
484*10465441SEvalZero 		wmove(win, y + i, x);
485*10465441SEvalZero 		for (j = 0; j < width; j++)
486*10465441SEvalZero 			if (!i && !j)
487*10465441SEvalZero 				waddch(win, border | ACS_ULCORNER);
488*10465441SEvalZero 			else if (i == height - 1 && !j)
489*10465441SEvalZero 				waddch(win, border | ACS_LLCORNER);
490*10465441SEvalZero 			else if (!i && j == width - 1)
491*10465441SEvalZero 				waddch(win, box | ACS_URCORNER);
492*10465441SEvalZero 			else if (i == height - 1 && j == width - 1)
493*10465441SEvalZero 				waddch(win, box | ACS_LRCORNER);
494*10465441SEvalZero 			else if (!i)
495*10465441SEvalZero 				waddch(win, border | ACS_HLINE);
496*10465441SEvalZero 			else if (i == height - 1)
497*10465441SEvalZero 				waddch(win, box | ACS_HLINE);
498*10465441SEvalZero 			else if (!j)
499*10465441SEvalZero 				waddch(win, border | ACS_VLINE);
500*10465441SEvalZero 			else if (j == width - 1)
501*10465441SEvalZero 				waddch(win, box | ACS_VLINE);
502*10465441SEvalZero 			else
503*10465441SEvalZero 				waddch(win, box | ' ');
504*10465441SEvalZero 	}
505*10465441SEvalZero }
506*10465441SEvalZero 
507*10465441SEvalZero /*
508*10465441SEvalZero  * Draw shadows along the right and bottom edge to give a more 3D look
509*10465441SEvalZero  * to the boxes
510*10465441SEvalZero  */
draw_shadow(WINDOW * win,int y,int x,int height,int width)511*10465441SEvalZero void draw_shadow(WINDOW * win, int y, int x, int height, int width)
512*10465441SEvalZero {
513*10465441SEvalZero 	int i;
514*10465441SEvalZero 
515*10465441SEvalZero 	if (has_colors()) {	/* Whether terminal supports color? */
516*10465441SEvalZero 		wattrset(win, dlg.shadow.atr);
517*10465441SEvalZero 		wmove(win, y + height, x + 2);
518*10465441SEvalZero 		for (i = 0; i < width; i++)
519*10465441SEvalZero 			waddch(win, winch(win) & A_CHARTEXT);
520*10465441SEvalZero 		for (i = y + 1; i < y + height + 1; i++) {
521*10465441SEvalZero 			wmove(win, i, x + width);
522*10465441SEvalZero 			waddch(win, winch(win) & A_CHARTEXT);
523*10465441SEvalZero 			waddch(win, winch(win) & A_CHARTEXT);
524*10465441SEvalZero 		}
525*10465441SEvalZero 		wnoutrefresh(win);
526*10465441SEvalZero 	}
527*10465441SEvalZero }
528*10465441SEvalZero 
529*10465441SEvalZero /*
530*10465441SEvalZero  *  Return the position of the first alphabetic character in a string.
531*10465441SEvalZero  */
first_alpha(const char * string,const char * exempt)532*10465441SEvalZero int first_alpha(const char *string, const char *exempt)
533*10465441SEvalZero {
534*10465441SEvalZero 	int i, in_paren = 0, c;
535*10465441SEvalZero 
536*10465441SEvalZero 	for (i = 0; i < strlen(string); i++) {
537*10465441SEvalZero 		c = tolower(string[i]);
538*10465441SEvalZero 
539*10465441SEvalZero 		if (strchr("<[(", c))
540*10465441SEvalZero 			++in_paren;
541*10465441SEvalZero 		if (strchr(">])", c) && in_paren > 0)
542*10465441SEvalZero 			--in_paren;
543*10465441SEvalZero 
544*10465441SEvalZero 		if ((!in_paren) && isalpha(c) && strchr(exempt, c) == 0)
545*10465441SEvalZero 			return i;
546*10465441SEvalZero 	}
547*10465441SEvalZero 
548*10465441SEvalZero 	return 0;
549*10465441SEvalZero }
550*10465441SEvalZero 
551*10465441SEvalZero /*
552*10465441SEvalZero  * ncurses uses ESC to detect escaped char sequences. This resutl in
553*10465441SEvalZero  * a small timeout before ESC is actually delivered to the application.
554*10465441SEvalZero  * lxdialog suggest <ESC> <ESC> which is correctly translated to two
555*10465441SEvalZero  * times esc. But then we need to ignore the second esc to avoid stepping
556*10465441SEvalZero  * out one menu too much. Filter away all escaped key sequences since
557*10465441SEvalZero  * keypad(FALSE) turn off ncurses support for escape sequences - and thats
558*10465441SEvalZero  * needed to make notimeout() do as expected.
559*10465441SEvalZero  */
on_key_esc(WINDOW * win)560*10465441SEvalZero int on_key_esc(WINDOW *win)
561*10465441SEvalZero {
562*10465441SEvalZero 	int key;
563*10465441SEvalZero 	int key2;
564*10465441SEvalZero 	int key3;
565*10465441SEvalZero 
566*10465441SEvalZero 	nodelay(win, TRUE);
567*10465441SEvalZero 	keypad(win, FALSE);
568*10465441SEvalZero 	key = wgetch(win);
569*10465441SEvalZero 	key2 = wgetch(win);
570*10465441SEvalZero 	do {
571*10465441SEvalZero 		key3 = wgetch(win);
572*10465441SEvalZero 	} while (key3 != ERR);
573*10465441SEvalZero 	nodelay(win, FALSE);
574*10465441SEvalZero 	keypad(win, TRUE);
575*10465441SEvalZero 	if (key == KEY_ESC && key2 == ERR)
576*10465441SEvalZero 		return KEY_ESC;
577*10465441SEvalZero 	else if (key != ERR && key != KEY_ESC && key2 == ERR)
578*10465441SEvalZero 		ungetch(key);
579*10465441SEvalZero 
580*10465441SEvalZero 	return -1;
581*10465441SEvalZero }
582*10465441SEvalZero 
583*10465441SEvalZero /* redraw screen in new size */
on_key_resize(void)584*10465441SEvalZero int on_key_resize(void)
585*10465441SEvalZero {
586*10465441SEvalZero 	dialog_clear();
587*10465441SEvalZero 	return KEY_RESIZE;
588*10465441SEvalZero }
589*10465441SEvalZero 
590*10465441SEvalZero struct dialog_list *item_cur;
591*10465441SEvalZero struct dialog_list item_nil;
592*10465441SEvalZero struct dialog_list *item_head;
593*10465441SEvalZero 
item_reset(void)594*10465441SEvalZero void item_reset(void)
595*10465441SEvalZero {
596*10465441SEvalZero 	struct dialog_list *p, *next;
597*10465441SEvalZero 
598*10465441SEvalZero 	for (p = item_head; p; p = next) {
599*10465441SEvalZero 		next = p->next;
600*10465441SEvalZero 		free(p);
601*10465441SEvalZero 	}
602*10465441SEvalZero 	item_head = NULL;
603*10465441SEvalZero 	item_cur = &item_nil;
604*10465441SEvalZero }
605*10465441SEvalZero 
item_make(const char * fmt,...)606*10465441SEvalZero void item_make(const char *fmt, ...)
607*10465441SEvalZero {
608*10465441SEvalZero 	va_list ap;
609*10465441SEvalZero 	struct dialog_list *p = malloc(sizeof(*p));
610*10465441SEvalZero 
611*10465441SEvalZero 	if (item_head)
612*10465441SEvalZero 		item_cur->next = p;
613*10465441SEvalZero 	else
614*10465441SEvalZero 		item_head = p;
615*10465441SEvalZero 	item_cur = p;
616*10465441SEvalZero 	memset(p, 0, sizeof(*p));
617*10465441SEvalZero 
618*10465441SEvalZero 	va_start(ap, fmt);
619*10465441SEvalZero 	vsnprintf(item_cur->node.str, sizeof(item_cur->node.str), fmt, ap);
620*10465441SEvalZero 	va_end(ap);
621*10465441SEvalZero }
622*10465441SEvalZero 
item_add_str(const char * fmt,...)623*10465441SEvalZero void item_add_str(const char *fmt, ...)
624*10465441SEvalZero {
625*10465441SEvalZero 	va_list ap;
626*10465441SEvalZero 	size_t avail;
627*10465441SEvalZero 
628*10465441SEvalZero 	avail = sizeof(item_cur->node.str) - strlen(item_cur->node.str);
629*10465441SEvalZero 
630*10465441SEvalZero 	va_start(ap, fmt);
631*10465441SEvalZero 	vsnprintf(item_cur->node.str + strlen(item_cur->node.str),
632*10465441SEvalZero 		  avail, fmt, ap);
633*10465441SEvalZero 	item_cur->node.str[sizeof(item_cur->node.str) - 1] = '\0';
634*10465441SEvalZero 	va_end(ap);
635*10465441SEvalZero }
636*10465441SEvalZero 
item_set_tag(char tag)637*10465441SEvalZero void item_set_tag(char tag)
638*10465441SEvalZero {
639*10465441SEvalZero 	item_cur->node.tag = tag;
640*10465441SEvalZero }
item_set_data(void * ptr)641*10465441SEvalZero void item_set_data(void *ptr)
642*10465441SEvalZero {
643*10465441SEvalZero 	item_cur->node.data = ptr;
644*10465441SEvalZero }
645*10465441SEvalZero 
item_set_selected(int val)646*10465441SEvalZero void item_set_selected(int val)
647*10465441SEvalZero {
648*10465441SEvalZero 	item_cur->node.selected = val;
649*10465441SEvalZero }
650*10465441SEvalZero 
item_activate_selected(void)651*10465441SEvalZero int item_activate_selected(void)
652*10465441SEvalZero {
653*10465441SEvalZero 	item_foreach()
654*10465441SEvalZero 		if (item_is_selected())
655*10465441SEvalZero 			return 1;
656*10465441SEvalZero 	return 0;
657*10465441SEvalZero }
658*10465441SEvalZero 
item_data(void)659*10465441SEvalZero void *item_data(void)
660*10465441SEvalZero {
661*10465441SEvalZero 	return item_cur->node.data;
662*10465441SEvalZero }
663*10465441SEvalZero 
item_tag(void)664*10465441SEvalZero char item_tag(void)
665*10465441SEvalZero {
666*10465441SEvalZero 	return item_cur->node.tag;
667*10465441SEvalZero }
668*10465441SEvalZero 
item_count(void)669*10465441SEvalZero int item_count(void)
670*10465441SEvalZero {
671*10465441SEvalZero 	int n = 0;
672*10465441SEvalZero 	struct dialog_list *p;
673*10465441SEvalZero 
674*10465441SEvalZero 	for (p = item_head; p; p = p->next)
675*10465441SEvalZero 		n++;
676*10465441SEvalZero 	return n;
677*10465441SEvalZero }
678*10465441SEvalZero 
item_set(int n)679*10465441SEvalZero void item_set(int n)
680*10465441SEvalZero {
681*10465441SEvalZero 	int i = 0;
682*10465441SEvalZero 	item_foreach()
683*10465441SEvalZero 		if (i++ == n)
684*10465441SEvalZero 			return;
685*10465441SEvalZero }
686*10465441SEvalZero 
item_n(void)687*10465441SEvalZero int item_n(void)
688*10465441SEvalZero {
689*10465441SEvalZero 	int n = 0;
690*10465441SEvalZero 	struct dialog_list *p;
691*10465441SEvalZero 
692*10465441SEvalZero 	for (p = item_head; p; p = p->next) {
693*10465441SEvalZero 		if (p == item_cur)
694*10465441SEvalZero 			return n;
695*10465441SEvalZero 		n++;
696*10465441SEvalZero 	}
697*10465441SEvalZero 	return 0;
698*10465441SEvalZero }
699*10465441SEvalZero 
item_str(void)700*10465441SEvalZero const char *item_str(void)
701*10465441SEvalZero {
702*10465441SEvalZero 	return item_cur->node.str;
703*10465441SEvalZero }
704*10465441SEvalZero 
item_is_selected(void)705*10465441SEvalZero int item_is_selected(void)
706*10465441SEvalZero {
707*10465441SEvalZero 	return (item_cur->node.selected != 0);
708*10465441SEvalZero }
709*10465441SEvalZero 
item_is_tag(char tag)710*10465441SEvalZero int item_is_tag(char tag)
711*10465441SEvalZero {
712*10465441SEvalZero 	return (item_cur->node.tag == tag);
713*10465441SEvalZero }
714