xref: /nrf52832-nimble/rt-thread/tools/kconfig-frontends/libs/lxdialog/inputbox.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  *  inputbox.c -- implements the input box
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 "dialog.h"
23*10465441SEvalZero 
24*10465441SEvalZero char dialog_input_result[MAX_LEN + 1];
25*10465441SEvalZero 
26*10465441SEvalZero /*
27*10465441SEvalZero  *  Print the termination buttons
28*10465441SEvalZero  */
print_buttons(WINDOW * dialog,int height,int width,int selected)29*10465441SEvalZero static void print_buttons(WINDOW * dialog, int height, int width, int selected)
30*10465441SEvalZero {
31*10465441SEvalZero 	int x = width / 2 - 11;
32*10465441SEvalZero 	int y = height - 2;
33*10465441SEvalZero 
34*10465441SEvalZero 	print_button(dialog, gettext("  Ok  "), y, x, selected == 0);
35*10465441SEvalZero 	print_button(dialog, gettext(" Help "), y, x + 14, selected == 1);
36*10465441SEvalZero 
37*10465441SEvalZero 	wmove(dialog, y, x + 1 + 14 * selected);
38*10465441SEvalZero 	wrefresh(dialog);
39*10465441SEvalZero }
40*10465441SEvalZero 
41*10465441SEvalZero /*
42*10465441SEvalZero  * Display a dialog box for inputing a string
43*10465441SEvalZero  */
dialog_inputbox(const char * title,const char * prompt,int height,int width,const char * init)44*10465441SEvalZero int dialog_inputbox(const char *title, const char *prompt, int height, int width,
45*10465441SEvalZero 		    const char *init)
46*10465441SEvalZero {
47*10465441SEvalZero 	int i, x, y, box_y, box_x, box_width;
48*10465441SEvalZero 	int input_x = 0, key = 0, button = -1;
49*10465441SEvalZero 	int show_x, len, pos;
50*10465441SEvalZero 	char *instr = dialog_input_result;
51*10465441SEvalZero 	WINDOW *dialog;
52*10465441SEvalZero 
53*10465441SEvalZero 	if (!init)
54*10465441SEvalZero 		instr[0] = '\0';
55*10465441SEvalZero 	else
56*10465441SEvalZero 		strcpy(instr, init);
57*10465441SEvalZero 
58*10465441SEvalZero do_resize:
59*10465441SEvalZero 	if (getmaxy(stdscr) <= (height - INPUTBOX_HEIGTH_MIN))
60*10465441SEvalZero 		return -ERRDISPLAYTOOSMALL;
61*10465441SEvalZero 	if (getmaxx(stdscr) <= (width - INPUTBOX_WIDTH_MIN))
62*10465441SEvalZero 		return -ERRDISPLAYTOOSMALL;
63*10465441SEvalZero 
64*10465441SEvalZero 	/* center dialog box on screen */
65*10465441SEvalZero 	x = (getmaxx(stdscr) - width) / 2;
66*10465441SEvalZero 	y = (getmaxy(stdscr) - height) / 2;
67*10465441SEvalZero 
68*10465441SEvalZero 	draw_shadow(stdscr, y, x, height, width);
69*10465441SEvalZero 
70*10465441SEvalZero 	dialog = newwin(height, width, y, x);
71*10465441SEvalZero 	keypad(dialog, TRUE);
72*10465441SEvalZero 
73*10465441SEvalZero 	draw_box(dialog, 0, 0, height, width,
74*10465441SEvalZero 		 dlg.dialog.atr, dlg.border.atr);
75*10465441SEvalZero 	wattrset(dialog, dlg.border.atr);
76*10465441SEvalZero 	mvwaddch(dialog, height - 3, 0, ACS_LTEE);
77*10465441SEvalZero 	for (i = 0; i < width - 2; i++)
78*10465441SEvalZero 		waddch(dialog, ACS_HLINE);
79*10465441SEvalZero 	wattrset(dialog, dlg.dialog.atr);
80*10465441SEvalZero 	waddch(dialog, ACS_RTEE);
81*10465441SEvalZero 
82*10465441SEvalZero 	print_title(dialog, title, width);
83*10465441SEvalZero 
84*10465441SEvalZero 	wattrset(dialog, dlg.dialog.atr);
85*10465441SEvalZero 	print_autowrap(dialog, prompt, width - 2, 1, 3);
86*10465441SEvalZero 
87*10465441SEvalZero 	/* Draw the input field box */
88*10465441SEvalZero 	box_width = width - 6;
89*10465441SEvalZero 	getyx(dialog, y, x);
90*10465441SEvalZero 	box_y = y + 2;
91*10465441SEvalZero 	box_x = (width - box_width) / 2;
92*10465441SEvalZero 	draw_box(dialog, y + 1, box_x - 1, 3, box_width + 2,
93*10465441SEvalZero 		 dlg.dialog.atr, dlg.border.atr);
94*10465441SEvalZero 
95*10465441SEvalZero 	print_buttons(dialog, height, width, 0);
96*10465441SEvalZero 
97*10465441SEvalZero 	/* Set up the initial value */
98*10465441SEvalZero 	wmove(dialog, box_y, box_x);
99*10465441SEvalZero 	wattrset(dialog, dlg.inputbox.atr);
100*10465441SEvalZero 
101*10465441SEvalZero 	len = strlen(instr);
102*10465441SEvalZero 	pos = len;
103*10465441SEvalZero 
104*10465441SEvalZero 	if (len >= box_width) {
105*10465441SEvalZero 		show_x = len - box_width + 1;
106*10465441SEvalZero 		input_x = box_width - 1;
107*10465441SEvalZero 		for (i = 0; i < box_width - 1; i++)
108*10465441SEvalZero 			waddch(dialog, instr[show_x + i]);
109*10465441SEvalZero 	} else {
110*10465441SEvalZero 		show_x = 0;
111*10465441SEvalZero 		input_x = len;
112*10465441SEvalZero 		waddstr(dialog, instr);
113*10465441SEvalZero 	}
114*10465441SEvalZero 
115*10465441SEvalZero 	wmove(dialog, box_y, box_x + input_x);
116*10465441SEvalZero 
117*10465441SEvalZero 	wrefresh(dialog);
118*10465441SEvalZero 
119*10465441SEvalZero 	while (key != KEY_ESC) {
120*10465441SEvalZero 		key = wgetch(dialog);
121*10465441SEvalZero 
122*10465441SEvalZero 		if (button == -1) {	/* Input box selected */
123*10465441SEvalZero 			switch (key) {
124*10465441SEvalZero 			case TAB:
125*10465441SEvalZero 			case KEY_UP:
126*10465441SEvalZero 			case KEY_DOWN:
127*10465441SEvalZero 				break;
128*10465441SEvalZero 			case KEY_BACKSPACE:
129*10465441SEvalZero 			case 127:
130*10465441SEvalZero 				if (pos) {
131*10465441SEvalZero 					wattrset(dialog, dlg.inputbox.atr);
132*10465441SEvalZero 					if (input_x == 0) {
133*10465441SEvalZero 						show_x--;
134*10465441SEvalZero 					} else
135*10465441SEvalZero 						input_x--;
136*10465441SEvalZero 
137*10465441SEvalZero 					if (pos < len) {
138*10465441SEvalZero 						for (i = pos - 1; i < len; i++) {
139*10465441SEvalZero 							instr[i] = instr[i+1];
140*10465441SEvalZero 						}
141*10465441SEvalZero 					}
142*10465441SEvalZero 
143*10465441SEvalZero 					pos--;
144*10465441SEvalZero 					len--;
145*10465441SEvalZero 					instr[len] = '\0';
146*10465441SEvalZero 					wmove(dialog, box_y, box_x);
147*10465441SEvalZero 					for (i = 0; i < box_width; i++) {
148*10465441SEvalZero 						if (!instr[show_x + i]) {
149*10465441SEvalZero 							waddch(dialog, ' ');
150*10465441SEvalZero 							break;
151*10465441SEvalZero 						}
152*10465441SEvalZero 						waddch(dialog, instr[show_x + i]);
153*10465441SEvalZero 					}
154*10465441SEvalZero 					wmove(dialog, box_y, input_x + box_x);
155*10465441SEvalZero 					wrefresh(dialog);
156*10465441SEvalZero 				}
157*10465441SEvalZero 				continue;
158*10465441SEvalZero 			case KEY_LEFT:
159*10465441SEvalZero 				if (pos > 0) {
160*10465441SEvalZero 					if (input_x > 0) {
161*10465441SEvalZero 						wmove(dialog, box_y, --input_x + box_x);
162*10465441SEvalZero 					} else if (input_x == 0) {
163*10465441SEvalZero 						show_x--;
164*10465441SEvalZero 						wmove(dialog, box_y, box_x);
165*10465441SEvalZero 						for (i = 0; i < box_width; i++) {
166*10465441SEvalZero 							if (!instr[show_x + i]) {
167*10465441SEvalZero 								waddch(dialog, ' ');
168*10465441SEvalZero 								break;
169*10465441SEvalZero 							}
170*10465441SEvalZero 							waddch(dialog, instr[show_x + i]);
171*10465441SEvalZero 						}
172*10465441SEvalZero 						wmove(dialog, box_y, box_x);
173*10465441SEvalZero 					}
174*10465441SEvalZero 					pos--;
175*10465441SEvalZero 				}
176*10465441SEvalZero 				continue;
177*10465441SEvalZero 			case KEY_RIGHT:
178*10465441SEvalZero 				if (pos < len) {
179*10465441SEvalZero 					if (input_x < box_width - 1) {
180*10465441SEvalZero 						wmove(dialog, box_y, ++input_x + box_x);
181*10465441SEvalZero 					} else if (input_x == box_width - 1) {
182*10465441SEvalZero 						show_x++;
183*10465441SEvalZero 						wmove(dialog, box_y, box_x);
184*10465441SEvalZero 						for (i = 0; i < box_width; i++) {
185*10465441SEvalZero 							if (!instr[show_x + i]) {
186*10465441SEvalZero 								waddch(dialog, ' ');
187*10465441SEvalZero 								break;
188*10465441SEvalZero 							}
189*10465441SEvalZero 							waddch(dialog, instr[show_x + i]);
190*10465441SEvalZero 						}
191*10465441SEvalZero 						wmove(dialog, box_y, input_x + box_x);
192*10465441SEvalZero 					}
193*10465441SEvalZero 					pos++;
194*10465441SEvalZero 				}
195*10465441SEvalZero 				continue;
196*10465441SEvalZero 			default:
197*10465441SEvalZero 				if (key < 0x100 && isprint(key)) {
198*10465441SEvalZero 					if (len < MAX_LEN) {
199*10465441SEvalZero 						wattrset(dialog, dlg.inputbox.atr);
200*10465441SEvalZero 						if (pos < len) {
201*10465441SEvalZero 							for (i = len; i > pos; i--)
202*10465441SEvalZero 								instr[i] = instr[i-1];
203*10465441SEvalZero 							instr[pos] = key;
204*10465441SEvalZero 						} else {
205*10465441SEvalZero 							instr[len] = key;
206*10465441SEvalZero 						}
207*10465441SEvalZero 						pos++;
208*10465441SEvalZero 						len++;
209*10465441SEvalZero 						instr[len] = '\0';
210*10465441SEvalZero 
211*10465441SEvalZero 						if (input_x == box_width - 1) {
212*10465441SEvalZero 							show_x++;
213*10465441SEvalZero 						} else {
214*10465441SEvalZero 							input_x++;
215*10465441SEvalZero 						}
216*10465441SEvalZero 
217*10465441SEvalZero 						wmove(dialog, box_y, box_x);
218*10465441SEvalZero 						for (i = 0; i < box_width; i++) {
219*10465441SEvalZero 							if (!instr[show_x + i]) {
220*10465441SEvalZero 								waddch(dialog, ' ');
221*10465441SEvalZero 								break;
222*10465441SEvalZero 							}
223*10465441SEvalZero 							waddch(dialog, instr[show_x + i]);
224*10465441SEvalZero 						}
225*10465441SEvalZero 						wmove(dialog, box_y, input_x + box_x);
226*10465441SEvalZero 						wrefresh(dialog);
227*10465441SEvalZero 					} else
228*10465441SEvalZero 						flash();	/* Alarm user about overflow */
229*10465441SEvalZero 					continue;
230*10465441SEvalZero 				}
231*10465441SEvalZero 			}
232*10465441SEvalZero 		}
233*10465441SEvalZero 		switch (key) {
234*10465441SEvalZero 		case 'O':
235*10465441SEvalZero 		case 'o':
236*10465441SEvalZero 			delwin(dialog);
237*10465441SEvalZero 			return 0;
238*10465441SEvalZero 		case 'H':
239*10465441SEvalZero 		case 'h':
240*10465441SEvalZero 			delwin(dialog);
241*10465441SEvalZero 			return 1;
242*10465441SEvalZero 		case KEY_UP:
243*10465441SEvalZero 		case KEY_LEFT:
244*10465441SEvalZero 			switch (button) {
245*10465441SEvalZero 			case -1:
246*10465441SEvalZero 				button = 1;	/* Indicates "Help" button is selected */
247*10465441SEvalZero 				print_buttons(dialog, height, width, 1);
248*10465441SEvalZero 				break;
249*10465441SEvalZero 			case 0:
250*10465441SEvalZero 				button = -1;	/* Indicates input box is selected */
251*10465441SEvalZero 				print_buttons(dialog, height, width, 0);
252*10465441SEvalZero 				wmove(dialog, box_y, box_x + input_x);
253*10465441SEvalZero 				wrefresh(dialog);
254*10465441SEvalZero 				break;
255*10465441SEvalZero 			case 1:
256*10465441SEvalZero 				button = 0;	/* Indicates "OK" button is selected */
257*10465441SEvalZero 				print_buttons(dialog, height, width, 0);
258*10465441SEvalZero 				break;
259*10465441SEvalZero 			}
260*10465441SEvalZero 			break;
261*10465441SEvalZero 		case TAB:
262*10465441SEvalZero 		case KEY_DOWN:
263*10465441SEvalZero 		case KEY_RIGHT:
264*10465441SEvalZero 			switch (button) {
265*10465441SEvalZero 			case -1:
266*10465441SEvalZero 				button = 0;	/* Indicates "OK" button is selected */
267*10465441SEvalZero 				print_buttons(dialog, height, width, 0);
268*10465441SEvalZero 				break;
269*10465441SEvalZero 			case 0:
270*10465441SEvalZero 				button = 1;	/* Indicates "Help" button is selected */
271*10465441SEvalZero 				print_buttons(dialog, height, width, 1);
272*10465441SEvalZero 				break;
273*10465441SEvalZero 			case 1:
274*10465441SEvalZero 				button = -1;	/* Indicates input box is selected */
275*10465441SEvalZero 				print_buttons(dialog, height, width, 0);
276*10465441SEvalZero 				wmove(dialog, box_y, box_x + input_x);
277*10465441SEvalZero 				wrefresh(dialog);
278*10465441SEvalZero 				break;
279*10465441SEvalZero 			}
280*10465441SEvalZero 			break;
281*10465441SEvalZero 		case ' ':
282*10465441SEvalZero 		case '\n':
283*10465441SEvalZero 			delwin(dialog);
284*10465441SEvalZero 			return (button == -1 ? 0 : button);
285*10465441SEvalZero 		case 'X':
286*10465441SEvalZero 		case 'x':
287*10465441SEvalZero 			key = KEY_ESC;
288*10465441SEvalZero 			break;
289*10465441SEvalZero 		case KEY_ESC:
290*10465441SEvalZero 			key = on_key_esc(dialog);
291*10465441SEvalZero 			break;
292*10465441SEvalZero 		case KEY_RESIZE:
293*10465441SEvalZero 			delwin(dialog);
294*10465441SEvalZero 			on_key_resize();
295*10465441SEvalZero 			goto do_resize;
296*10465441SEvalZero 		}
297*10465441SEvalZero 	}
298*10465441SEvalZero 
299*10465441SEvalZero 	delwin(dialog);
300*10465441SEvalZero 	return KEY_ESC;		/* ESC pressed */
301*10465441SEvalZero }
302