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