Merge tag 'xtensa-20230905' of https://github.com/jcmvbkbc/linux-xtensa
[platform/kernel/linux-starfive.git] / scripts / kconfig / nconf.gui.c
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2008 Nir Tzachar <nir.tzachar@gmail.com>
4  *
5  * Derived from menuconfig.
6  */
7 #include "nconf.h"
8 #include "lkc.h"
9
10 int attr_normal;
11 int attr_main_heading;
12 int attr_main_menu_box;
13 int attr_main_menu_fore;
14 int attr_main_menu_back;
15 int attr_main_menu_grey;
16 int attr_main_menu_heading;
17 int attr_scrollwin_text;
18 int attr_scrollwin_heading;
19 int attr_scrollwin_box;
20 int attr_dialog_text;
21 int attr_dialog_menu_fore;
22 int attr_dialog_menu_back;
23 int attr_dialog_box;
24 int attr_input_box;
25 int attr_input_heading;
26 int attr_input_text;
27 int attr_input_field;
28 int attr_function_text;
29 int attr_function_highlight;
30
31 #define COLOR_ATTR(_at, _fg, _bg, _hl) \
32         { .attr = &(_at), .has_color = true, .color_fg = _fg, .color_bg = _bg, .highlight = _hl }
33 #define NO_COLOR_ATTR(_at, _hl) \
34         { .attr = &(_at), .has_color = false, .highlight = _hl }
35 #define COLOR_DEFAULT           -1
36
37 struct nconf_attr_param {
38         int *attr;
39         bool has_color;
40         int color_fg;
41         int color_bg;
42         int highlight;
43 };
44
45 static const struct nconf_attr_param color_theme_params[] = {
46         COLOR_ATTR(attr_normal,                 COLOR_DEFAULT,  COLOR_DEFAULT,  A_NORMAL),
47         COLOR_ATTR(attr_main_heading,           COLOR_MAGENTA,  COLOR_DEFAULT,  A_BOLD | A_UNDERLINE),
48         COLOR_ATTR(attr_main_menu_box,          COLOR_YELLOW,   COLOR_DEFAULT,  A_NORMAL),
49         COLOR_ATTR(attr_main_menu_fore,         COLOR_DEFAULT,  COLOR_DEFAULT,  A_REVERSE),
50         COLOR_ATTR(attr_main_menu_back,         COLOR_DEFAULT,  COLOR_DEFAULT,  A_NORMAL),
51         COLOR_ATTR(attr_main_menu_grey,         COLOR_DEFAULT,  COLOR_DEFAULT,  A_NORMAL),
52         COLOR_ATTR(attr_main_menu_heading,      COLOR_GREEN,    COLOR_DEFAULT,  A_BOLD),
53         COLOR_ATTR(attr_scrollwin_text,         COLOR_DEFAULT,  COLOR_DEFAULT,  A_NORMAL),
54         COLOR_ATTR(attr_scrollwin_heading,      COLOR_GREEN,    COLOR_DEFAULT,  A_BOLD),
55         COLOR_ATTR(attr_scrollwin_box,          COLOR_YELLOW,   COLOR_DEFAULT,  A_BOLD),
56         COLOR_ATTR(attr_dialog_text,            COLOR_DEFAULT,  COLOR_DEFAULT,  A_BOLD),
57         COLOR_ATTR(attr_dialog_menu_fore,       COLOR_RED,      COLOR_DEFAULT,  A_STANDOUT),
58         COLOR_ATTR(attr_dialog_menu_back,       COLOR_YELLOW,   COLOR_DEFAULT,  A_NORMAL),
59         COLOR_ATTR(attr_dialog_box,             COLOR_YELLOW,   COLOR_DEFAULT,  A_BOLD),
60         COLOR_ATTR(attr_input_box,              COLOR_YELLOW,   COLOR_DEFAULT,  A_NORMAL),
61         COLOR_ATTR(attr_input_heading,          COLOR_GREEN,    COLOR_DEFAULT,  A_BOLD),
62         COLOR_ATTR(attr_input_text,             COLOR_DEFAULT,  COLOR_DEFAULT,  A_NORMAL),
63         COLOR_ATTR(attr_input_field,            COLOR_DEFAULT,  COLOR_DEFAULT,  A_UNDERLINE),
64         COLOR_ATTR(attr_function_text,          COLOR_YELLOW,   COLOR_DEFAULT,  A_REVERSE),
65         COLOR_ATTR(attr_function_highlight,     COLOR_DEFAULT,  COLOR_DEFAULT,  A_BOLD),
66         { /* sentinel */ }
67 };
68
69 static const struct nconf_attr_param no_color_theme_params[] = {
70         NO_COLOR_ATTR(attr_normal,              A_NORMAL),
71         NO_COLOR_ATTR(attr_main_heading,        A_BOLD | A_UNDERLINE),
72         NO_COLOR_ATTR(attr_main_menu_box,       A_NORMAL),
73         NO_COLOR_ATTR(attr_main_menu_fore,      A_STANDOUT),
74         NO_COLOR_ATTR(attr_main_menu_back,      A_NORMAL),
75         NO_COLOR_ATTR(attr_main_menu_grey,      A_NORMAL),
76         NO_COLOR_ATTR(attr_main_menu_heading,   A_BOLD),
77         NO_COLOR_ATTR(attr_scrollwin_text,      A_NORMAL),
78         NO_COLOR_ATTR(attr_scrollwin_heading,   A_BOLD),
79         NO_COLOR_ATTR(attr_scrollwin_box,       A_BOLD),
80         NO_COLOR_ATTR(attr_dialog_text,         A_NORMAL),
81         NO_COLOR_ATTR(attr_dialog_menu_fore,    A_STANDOUT),
82         NO_COLOR_ATTR(attr_dialog_menu_back,    A_NORMAL),
83         NO_COLOR_ATTR(attr_dialog_box,          A_BOLD),
84         NO_COLOR_ATTR(attr_input_box,           A_BOLD),
85         NO_COLOR_ATTR(attr_input_heading,       A_BOLD),
86         NO_COLOR_ATTR(attr_input_text,          A_NORMAL),
87         NO_COLOR_ATTR(attr_input_field,         A_UNDERLINE),
88         NO_COLOR_ATTR(attr_function_text,       A_REVERSE),
89         NO_COLOR_ATTR(attr_function_highlight,  A_BOLD),
90         { /* sentinel */ }
91 };
92
93 void set_colors(void)
94 {
95         const struct nconf_attr_param *p;
96         int pair = 0;
97
98         if (has_colors()) {
99                 start_color();
100                 use_default_colors();
101                 p = color_theme_params;
102         } else {
103                 p = no_color_theme_params;
104         }
105
106         for (; p->attr; p++) {
107                 int attr = p->highlight;
108
109                 if (p->has_color) {
110                         pair++;
111                         init_pair(pair, p->color_fg, p->color_bg);
112                         attr |= COLOR_PAIR(pair);
113                 }
114
115                 *p->attr = attr;
116         }
117 }
118
119 /* this changes the windows attributes !!! */
120 void print_in_middle(WINDOW *win, int y, int width, const char *str, int attrs)
121 {
122         wattrset(win, attrs);
123         mvwprintw(win, y, (width - strlen(str)) / 2, "%s", str);
124 }
125
126 int get_line_no(const char *text)
127 {
128         int i;
129         int total = 1;
130
131         if (!text)
132                 return 0;
133
134         for (i = 0; text[i] != '\0'; i++)
135                 if (text[i] == '\n')
136                         total++;
137         return total;
138 }
139
140 const char *get_line(const char *text, int line_no)
141 {
142         int i;
143         int lines = 0;
144
145         if (!text)
146                 return NULL;
147
148         for (i = 0; text[i] != '\0' && lines < line_no; i++)
149                 if (text[i] == '\n')
150                         lines++;
151         return text+i;
152 }
153
154 int get_line_length(const char *line)
155 {
156         int res = 0;
157         while (*line != '\0' && *line != '\n') {
158                 line++;
159                 res++;
160         }
161         return res;
162 }
163
164 /* print all lines to the window. */
165 void fill_window(WINDOW *win, const char *text)
166 {
167         int x, y;
168         int total_lines = get_line_no(text);
169         int i;
170
171         getmaxyx(win, y, x);
172         /* do not go over end of line */
173         total_lines = min(total_lines, y);
174         for (i = 0; i < total_lines; i++) {
175                 char tmp[x+10];
176                 const char *line = get_line(text, i);
177                 int len = get_line_length(line);
178                 strncpy(tmp, line, min(len, x));
179                 tmp[len] = '\0';
180                 mvwprintw(win, i, 0, "%s", tmp);
181         }
182 }
183
184 /* get the message, and buttons.
185  * each button must be a char*
186  * return the selected button
187  *
188  * this dialog is used for 2 different things:
189  * 1) show a text box, no buttons.
190  * 2) show a dialog, with horizontal buttons
191  */
192 int btn_dialog(WINDOW *main_window, const char *msg, int btn_num, ...)
193 {
194         va_list ap;
195         char *btn;
196         int btns_width = 0;
197         int msg_lines = 0;
198         int msg_width = 0;
199         int total_width;
200         int win_rows = 0;
201         WINDOW *win;
202         WINDOW *msg_win;
203         WINDOW *menu_win;
204         MENU *menu;
205         ITEM *btns[btn_num+1];
206         int i, x, y;
207         int res = -1;
208
209
210         va_start(ap, btn_num);
211         for (i = 0; i < btn_num; i++) {
212                 btn = va_arg(ap, char *);
213                 btns[i] = new_item(btn, "");
214                 btns_width += strlen(btn)+1;
215         }
216         va_end(ap);
217         btns[btn_num] = NULL;
218
219         /* find the widest line of msg: */
220         msg_lines = get_line_no(msg);
221         for (i = 0; i < msg_lines; i++) {
222                 const char *line = get_line(msg, i);
223                 int len = get_line_length(line);
224                 if (msg_width < len)
225                         msg_width = len;
226         }
227
228         total_width = max(msg_width, btns_width);
229         /* place dialog in middle of screen */
230         y = (getmaxy(stdscr)-(msg_lines+4))/2;
231         x = (getmaxx(stdscr)-(total_width+4))/2;
232
233
234         /* create the windows */
235         if (btn_num > 0)
236                 win_rows = msg_lines+4;
237         else
238                 win_rows = msg_lines+2;
239
240         win = newwin(win_rows, total_width+4, y, x);
241         keypad(win, TRUE);
242         menu_win = derwin(win, 1, btns_width, win_rows-2,
243                         1+(total_width+2-btns_width)/2);
244         menu = new_menu(btns);
245         msg_win = derwin(win, win_rows-2, msg_width, 1,
246                         1+(total_width+2-msg_width)/2);
247
248         set_menu_fore(menu, attr_dialog_menu_fore);
249         set_menu_back(menu, attr_dialog_menu_back);
250
251         wattrset(win, attr_dialog_box);
252         box(win, 0, 0);
253
254         /* print message */
255         wattrset(msg_win, attr_dialog_text);
256         fill_window(msg_win, msg);
257
258         set_menu_win(menu, win);
259         set_menu_sub(menu, menu_win);
260         set_menu_format(menu, 1, btn_num);
261         menu_opts_off(menu, O_SHOWDESC);
262         menu_opts_off(menu, O_SHOWMATCH);
263         menu_opts_on(menu, O_ONEVALUE);
264         menu_opts_on(menu, O_NONCYCLIC);
265         set_menu_mark(menu, "");
266         post_menu(menu);
267
268
269         touchwin(win);
270         refresh_all_windows(main_window);
271         while ((res = wgetch(win))) {
272                 switch (res) {
273                 case KEY_LEFT:
274                         menu_driver(menu, REQ_LEFT_ITEM);
275                         break;
276                 case KEY_RIGHT:
277                         menu_driver(menu, REQ_RIGHT_ITEM);
278                         break;
279                 case 10: /* ENTER */
280                 case 27: /* ESCAPE */
281                 case ' ':
282                 case KEY_F(F_BACK):
283                 case KEY_F(F_EXIT):
284                         break;
285                 }
286                 touchwin(win);
287                 refresh_all_windows(main_window);
288
289                 if (res == 10 || res == ' ') {
290                         res = item_index(current_item(menu));
291                         break;
292                 } else if (res == 27 || res == KEY_F(F_BACK) ||
293                                 res == KEY_F(F_EXIT)) {
294                         res = KEY_EXIT;
295                         break;
296                 }
297         }
298
299         unpost_menu(menu);
300         free_menu(menu);
301         for (i = 0; i < btn_num; i++)
302                 free_item(btns[i]);
303
304         delwin(win);
305         return res;
306 }
307
308 int dialog_inputbox(WINDOW *main_window,
309                 const char *title, const char *prompt,
310                 const char *init, char **resultp, int *result_len)
311 {
312         int prompt_lines = 0;
313         int prompt_width = 0;
314         WINDOW *win;
315         WINDOW *prompt_win;
316         WINDOW *form_win;
317         PANEL *panel;
318         int i, x, y, lines, columns, win_lines, win_cols;
319         int res = -1;
320         int cursor_position = strlen(init);
321         int cursor_form_win;
322         char *result = *resultp;
323
324         getmaxyx(stdscr, lines, columns);
325
326         if (strlen(init)+1 > *result_len) {
327                 *result_len = strlen(init)+1;
328                 *resultp = result = xrealloc(result, *result_len);
329         }
330
331         /* find the widest line of msg: */
332         prompt_lines = get_line_no(prompt);
333         for (i = 0; i < prompt_lines; i++) {
334                 const char *line = get_line(prompt, i);
335                 int len = get_line_length(line);
336                 prompt_width = max(prompt_width, len);
337         }
338
339         if (title)
340                 prompt_width = max(prompt_width, strlen(title));
341
342         win_lines = min(prompt_lines+6, lines-2);
343         win_cols = min(prompt_width+7, columns-2);
344         prompt_lines = max(win_lines-6, 0);
345         prompt_width = max(win_cols-7, 0);
346
347         /* place dialog in middle of screen */
348         y = (lines-win_lines)/2;
349         x = (columns-win_cols)/2;
350
351         strncpy(result, init, *result_len);
352
353         /* create the windows */
354         win = newwin(win_lines, win_cols, y, x);
355         prompt_win = derwin(win, prompt_lines+1, prompt_width, 2, 2);
356         form_win = derwin(win, 1, prompt_width, prompt_lines+3, 2);
357         keypad(form_win, TRUE);
358
359         wattrset(form_win, attr_input_field);
360
361         wattrset(win, attr_input_box);
362         box(win, 0, 0);
363         wattrset(win, attr_input_heading);
364         if (title)
365                 mvwprintw(win, 0, 3, "%s", title);
366
367         /* print message */
368         wattrset(prompt_win, attr_input_text);
369         fill_window(prompt_win, prompt);
370
371         mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
372         cursor_form_win = min(cursor_position, prompt_width-1);
373         mvwprintw(form_win, 0, 0, "%s",
374                   result + cursor_position-cursor_form_win);
375
376         /* create panels */
377         panel = new_panel(win);
378
379         /* show the cursor */
380         curs_set(1);
381
382         touchwin(win);
383         refresh_all_windows(main_window);
384         while ((res = wgetch(form_win))) {
385                 int len = strlen(result);
386                 switch (res) {
387                 case 10: /* ENTER */
388                 case 27: /* ESCAPE */
389                 case KEY_F(F_HELP):
390                 case KEY_F(F_EXIT):
391                 case KEY_F(F_BACK):
392                         break;
393                 case 8:   /* ^H */
394                 case 127: /* ^? */
395                 case KEY_BACKSPACE:
396                         if (cursor_position > 0) {
397                                 memmove(&result[cursor_position-1],
398                                                 &result[cursor_position],
399                                                 len-cursor_position+1);
400                                 cursor_position--;
401                                 cursor_form_win--;
402                                 len--;
403                         }
404                         break;
405                 case KEY_DC:
406                         if (cursor_position >= 0 && cursor_position < len) {
407                                 memmove(&result[cursor_position],
408                                                 &result[cursor_position+1],
409                                                 len-cursor_position+1);
410                                 len--;
411                         }
412                         break;
413                 case KEY_UP:
414                 case KEY_RIGHT:
415                         if (cursor_position < len) {
416                                 cursor_position++;
417                                 cursor_form_win++;
418                         }
419                         break;
420                 case KEY_DOWN:
421                 case KEY_LEFT:
422                         if (cursor_position > 0) {
423                                 cursor_position--;
424                                 cursor_form_win--;
425                         }
426                         break;
427                 case KEY_HOME:
428                         cursor_position = 0;
429                         cursor_form_win = 0;
430                         break;
431                 case KEY_END:
432                         cursor_position = len;
433                         cursor_form_win = min(cursor_position, prompt_width-1);
434                         break;
435                 default:
436                         if ((isgraph(res) || isspace(res))) {
437                                 /* one for new char, one for '\0' */
438                                 if (len+2 > *result_len) {
439                                         *result_len = len+2;
440                                         *resultp = result = realloc(result,
441                                                                 *result_len);
442                                 }
443                                 /* insert the char at the proper position */
444                                 memmove(&result[cursor_position+1],
445                                                 &result[cursor_position],
446                                                 len-cursor_position+1);
447                                 result[cursor_position] = res;
448                                 cursor_position++;
449                                 cursor_form_win++;
450                                 len++;
451                         } else {
452                                 mvprintw(0, 0, "unknown key: %d\n", res);
453                         }
454                         break;
455                 }
456                 if (cursor_form_win < 0)
457                         cursor_form_win = 0;
458                 else if (cursor_form_win > prompt_width-1)
459                         cursor_form_win = prompt_width-1;
460
461                 wmove(form_win, 0, 0);
462                 wclrtoeol(form_win);
463                 mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
464                 mvwprintw(form_win, 0, 0, "%s",
465                         result + cursor_position-cursor_form_win);
466                 wmove(form_win, 0, cursor_form_win);
467                 touchwin(win);
468                 refresh_all_windows(main_window);
469
470                 if (res == 10) {
471                         res = 0;
472                         break;
473                 } else if (res == 27 || res == KEY_F(F_BACK) ||
474                                 res == KEY_F(F_EXIT)) {
475                         res = KEY_EXIT;
476                         break;
477                 } else if (res == KEY_F(F_HELP)) {
478                         res = 1;
479                         break;
480                 }
481         }
482
483         /* hide the cursor */
484         curs_set(0);
485         del_panel(panel);
486         delwin(prompt_win);
487         delwin(form_win);
488         delwin(win);
489         return res;
490 }
491
492 /* refresh all windows in the correct order */
493 void refresh_all_windows(WINDOW *main_window)
494 {
495         update_panels();
496         touchwin(main_window);
497         refresh();
498 }
499
500 void show_scroll_win(WINDOW *main_window,
501                 const char *title,
502                 const char *text)
503 {
504         (void)show_scroll_win_ext(main_window, title, (char *)text, NULL, NULL, NULL, NULL);
505 }
506
507 /* layman's scrollable window... */
508 int show_scroll_win_ext(WINDOW *main_window, const char *title, char *text,
509                         int *vscroll, int *hscroll,
510                         extra_key_cb_fn extra_key_cb, void *data)
511 {
512         int res;
513         int total_lines = get_line_no(text);
514         int x, y, lines, columns;
515         int start_x = 0, start_y = 0;
516         int text_lines = 0, text_cols = 0;
517         int total_cols = 0;
518         int win_cols = 0;
519         int win_lines = 0;
520         int i = 0;
521         WINDOW *win;
522         WINDOW *pad;
523         PANEL *panel;
524         bool done = false;
525
526         if (hscroll)
527                 start_x = *hscroll;
528         if (vscroll)
529                 start_y = *vscroll;
530
531         getmaxyx(stdscr, lines, columns);
532
533         /* find the widest line of msg: */
534         total_lines = get_line_no(text);
535         for (i = 0; i < total_lines; i++) {
536                 const char *line = get_line(text, i);
537                 int len = get_line_length(line);
538                 total_cols = max(total_cols, len+2);
539         }
540
541         /* create the pad */
542         pad = newpad(total_lines+10, total_cols+10);
543         wattrset(pad, attr_scrollwin_text);
544         fill_window(pad, text);
545
546         win_lines = min(total_lines+4, lines-2);
547         win_cols = min(total_cols+2, columns-2);
548         text_lines = max(win_lines-4, 0);
549         text_cols = max(win_cols-2, 0);
550
551         /* place window in middle of screen */
552         y = (lines-win_lines)/2;
553         x = (columns-win_cols)/2;
554
555         win = newwin(win_lines, win_cols, y, x);
556         keypad(win, TRUE);
557         /* show the help in the help window, and show the help panel */
558         wattrset(win, attr_scrollwin_box);
559         box(win, 0, 0);
560         wattrset(win, attr_scrollwin_heading);
561         mvwprintw(win, 0, 3, " %s ", title);
562         panel = new_panel(win);
563
564         /* handle scrolling */
565         while (!done) {
566                 copywin(pad, win, start_y, start_x, 2, 2, text_lines,
567                                 text_cols, 0);
568                 print_in_middle(win,
569                                 text_lines+2,
570                                 text_cols,
571                                 "<OK>",
572                                 attr_dialog_menu_fore);
573                 wrefresh(win);
574
575                 res = wgetch(win);
576                 switch (res) {
577                 case KEY_NPAGE:
578                 case ' ':
579                 case 'd':
580                         start_y += text_lines-2;
581                         break;
582                 case KEY_PPAGE:
583                 case 'u':
584                         start_y -= text_lines+2;
585                         break;
586                 case KEY_HOME:
587                         start_y = 0;
588                         break;
589                 case KEY_END:
590                         start_y = total_lines-text_lines;
591                         break;
592                 case KEY_DOWN:
593                 case 'j':
594                         start_y++;
595                         break;
596                 case KEY_UP:
597                 case 'k':
598                         start_y--;
599                         break;
600                 case KEY_LEFT:
601                 case 'h':
602                         start_x--;
603                         break;
604                 case KEY_RIGHT:
605                 case 'l':
606                         start_x++;
607                         break;
608                 default:
609                         if (extra_key_cb) {
610                                 size_t start = (get_line(text, start_y) - text);
611                                 size_t end = (get_line(text, start_y + text_lines) - text);
612
613                                 if (extra_key_cb(res, start, end, data)) {
614                                         done = true;
615                                         break;
616                                 }
617                         }
618                 }
619                 if (res == 0 || res == 10 || res == 27 || res == 'q' ||
620                         res == KEY_F(F_HELP) || res == KEY_F(F_BACK) ||
621                         res == KEY_F(F_EXIT))
622                         break;
623                 if (start_y < 0)
624                         start_y = 0;
625                 if (start_y >= total_lines-text_lines)
626                         start_y = total_lines-text_lines;
627                 if (start_x < 0)
628                         start_x = 0;
629                 if (start_x >= total_cols-text_cols)
630                         start_x = total_cols-text_cols;
631         }
632
633         if (hscroll)
634                 *hscroll = start_x;
635         if (vscroll)
636                 *vscroll = start_y;
637         del_panel(panel);
638         delwin(win);
639         refresh_all_windows(main_window);
640         return res;
641 }