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