123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (C) 2008 Nir Tzachar <nir.tzachar@gmail.com>
- *
- * Derived from menuconfig.
- */
- #include "nconf.h"
- #include "lkc.h"
- int attr_normal;
- int attr_main_heading;
- int attr_main_menu_box;
- int attr_main_menu_fore;
- int attr_main_menu_back;
- int attr_main_menu_grey;
- int attr_main_menu_heading;
- int attr_scrollwin_text;
- int attr_scrollwin_heading;
- int attr_scrollwin_box;
- int attr_dialog_text;
- int attr_dialog_menu_fore;
- int attr_dialog_menu_back;
- int attr_dialog_box;
- int attr_input_box;
- int attr_input_heading;
- int attr_input_text;
- int attr_input_field;
- int attr_function_text;
- int attr_function_highlight;
- #define COLOR_ATTR(_at, _fg, _bg, _hl) \
- { .attr = &(_at), .has_color = true, .color_fg = _fg, .color_bg = _bg, .highlight = _hl }
- #define NO_COLOR_ATTR(_at, _hl) \
- { .attr = &(_at), .has_color = false, .highlight = _hl }
- #define COLOR_DEFAULT -1
- struct nconf_attr_param {
- int *attr;
- bool has_color;
- int color_fg;
- int color_bg;
- int highlight;
- };
- static const struct nconf_attr_param color_theme_params[] = {
- COLOR_ATTR(attr_normal, COLOR_DEFAULT, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_main_heading, COLOR_MAGENTA, COLOR_DEFAULT, A_BOLD | A_UNDERLINE),
- COLOR_ATTR(attr_main_menu_box, COLOR_YELLOW, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_main_menu_fore, COLOR_DEFAULT, COLOR_DEFAULT, A_REVERSE),
- COLOR_ATTR(attr_main_menu_back, COLOR_DEFAULT, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_main_menu_grey, COLOR_DEFAULT, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_main_menu_heading, COLOR_GREEN, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_scrollwin_text, COLOR_DEFAULT, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_scrollwin_heading, COLOR_GREEN, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_scrollwin_box, COLOR_YELLOW, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_dialog_text, COLOR_DEFAULT, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_dialog_menu_fore, COLOR_RED, COLOR_DEFAULT, A_STANDOUT),
- COLOR_ATTR(attr_dialog_menu_back, COLOR_YELLOW, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_dialog_box, COLOR_YELLOW, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_input_box, COLOR_YELLOW, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_input_heading, COLOR_GREEN, COLOR_DEFAULT, A_BOLD),
- COLOR_ATTR(attr_input_text, COLOR_DEFAULT, COLOR_DEFAULT, A_NORMAL),
- COLOR_ATTR(attr_input_field, COLOR_DEFAULT, COLOR_DEFAULT, A_UNDERLINE),
- COLOR_ATTR(attr_function_text, COLOR_YELLOW, COLOR_DEFAULT, A_REVERSE),
- COLOR_ATTR(attr_function_highlight, COLOR_DEFAULT, COLOR_DEFAULT, A_BOLD),
- { /* sentinel */ }
- };
- static const struct nconf_attr_param no_color_theme_params[] = {
- NO_COLOR_ATTR(attr_normal, A_NORMAL),
- NO_COLOR_ATTR(attr_main_heading, A_BOLD | A_UNDERLINE),
- NO_COLOR_ATTR(attr_main_menu_box, A_NORMAL),
- NO_COLOR_ATTR(attr_main_menu_fore, A_STANDOUT),
- NO_COLOR_ATTR(attr_main_menu_back, A_NORMAL),
- NO_COLOR_ATTR(attr_main_menu_grey, A_NORMAL),
- NO_COLOR_ATTR(attr_main_menu_heading, A_BOLD),
- NO_COLOR_ATTR(attr_scrollwin_text, A_NORMAL),
- NO_COLOR_ATTR(attr_scrollwin_heading, A_BOLD),
- NO_COLOR_ATTR(attr_scrollwin_box, A_BOLD),
- NO_COLOR_ATTR(attr_dialog_text, A_NORMAL),
- NO_COLOR_ATTR(attr_dialog_menu_fore, A_STANDOUT),
- NO_COLOR_ATTR(attr_dialog_menu_back, A_NORMAL),
- NO_COLOR_ATTR(attr_dialog_box, A_BOLD),
- NO_COLOR_ATTR(attr_input_box, A_BOLD),
- NO_COLOR_ATTR(attr_input_heading, A_BOLD),
- NO_COLOR_ATTR(attr_input_text, A_NORMAL),
- NO_COLOR_ATTR(attr_input_field, A_UNDERLINE),
- NO_COLOR_ATTR(attr_function_text, A_REVERSE),
- NO_COLOR_ATTR(attr_function_highlight, A_BOLD),
- { /* sentinel */ }
- };
- void set_colors(void)
- {
- const struct nconf_attr_param *p;
- int pair = 0;
- if (has_colors()) {
- start_color();
- use_default_colors();
- p = color_theme_params;
- } else {
- p = no_color_theme_params;
- }
- for (; p->attr; p++) {
- int attr = p->highlight;
- if (p->has_color) {
- pair++;
- init_pair(pair, p->color_fg, p->color_bg);
- attr |= COLOR_PAIR(pair);
- }
- *p->attr = attr;
- }
- }
- /* this changes the windows attributes !!! */
- void print_in_middle(WINDOW *win, int y, int width, const char *str, int attrs)
- {
- wattrset(win, attrs);
- mvwprintw(win, y, (width - strlen(str)) / 2, "%s", str);
- }
- int get_line_no(const char *text)
- {
- int i;
- int total = 1;
- if (!text)
- return 0;
- for (i = 0; text[i] != '\0'; i++)
- if (text[i] == '\n')
- total++;
- return total;
- }
- const char *get_line(const char *text, int line_no)
- {
- int i;
- int lines = 0;
- if (!text)
- return NULL;
- for (i = 0; text[i] != '\0' && lines < line_no; i++)
- if (text[i] == '\n')
- lines++;
- return text+i;
- }
- int get_line_length(const char *line)
- {
- int res = 0;
- while (*line != '\0' && *line != '\n') {
- line++;
- res++;
- }
- return res;
- }
- /* print all lines to the window. */
- void fill_window(WINDOW *win, const char *text)
- {
- int x, y;
- int total_lines = get_line_no(text);
- int i;
- getmaxyx(win, y, x);
- /* do not go over end of line */
- total_lines = min(total_lines, y);
- for (i = 0; i < total_lines; i++) {
- char tmp[x+10];
- const char *line = get_line(text, i);
- int len = get_line_length(line);
- strncpy(tmp, line, min(len, x));
- tmp[len] = '\0';
- mvwprintw(win, i, 0, "%s", tmp);
- }
- }
- /* get the message, and buttons.
- * each button must be a char*
- * return the selected button
- *
- * this dialog is used for 2 different things:
- * 1) show a text box, no buttons.
- * 2) show a dialog, with horizontal buttons
- */
- int btn_dialog(WINDOW *main_window, const char *msg, int btn_num, ...)
- {
- va_list ap;
- char *btn;
- int btns_width = 0;
- int msg_lines = 0;
- int msg_width = 0;
- int total_width;
- int win_rows = 0;
- WINDOW *win;
- WINDOW *msg_win;
- WINDOW *menu_win;
- MENU *menu;
- ITEM *btns[btn_num+1];
- int i, x, y;
- int res = -1;
- va_start(ap, btn_num);
- for (i = 0; i < btn_num; i++) {
- btn = va_arg(ap, char *);
- btns[i] = new_item(btn, "");
- btns_width += strlen(btn)+1;
- }
- va_end(ap);
- btns[btn_num] = NULL;
- /* find the widest line of msg: */
- msg_lines = get_line_no(msg);
- for (i = 0; i < msg_lines; i++) {
- const char *line = get_line(msg, i);
- int len = get_line_length(line);
- if (msg_width < len)
- msg_width = len;
- }
- total_width = max(msg_width, btns_width);
- /* place dialog in middle of screen */
- y = (getmaxy(stdscr)-(msg_lines+4))/2;
- x = (getmaxx(stdscr)-(total_width+4))/2;
- /* create the windows */
- if (btn_num > 0)
- win_rows = msg_lines+4;
- else
- win_rows = msg_lines+2;
- win = newwin(win_rows, total_width+4, y, x);
- keypad(win, TRUE);
- menu_win = derwin(win, 1, btns_width, win_rows-2,
- 1+(total_width+2-btns_width)/2);
- menu = new_menu(btns);
- msg_win = derwin(win, win_rows-2, msg_width, 1,
- 1+(total_width+2-msg_width)/2);
- set_menu_fore(menu, attr_dialog_menu_fore);
- set_menu_back(menu, attr_dialog_menu_back);
- wattrset(win, attr_dialog_box);
- box(win, 0, 0);
- /* print message */
- wattrset(msg_win, attr_dialog_text);
- fill_window(msg_win, msg);
- set_menu_win(menu, win);
- set_menu_sub(menu, menu_win);
- set_menu_format(menu, 1, btn_num);
- menu_opts_off(menu, O_SHOWDESC);
- menu_opts_off(menu, O_SHOWMATCH);
- menu_opts_on(menu, O_ONEVALUE);
- menu_opts_on(menu, O_NONCYCLIC);
- set_menu_mark(menu, "");
- post_menu(menu);
- touchwin(win);
- refresh_all_windows(main_window);
- while ((res = wgetch(win))) {
- switch (res) {
- case KEY_LEFT:
- menu_driver(menu, REQ_LEFT_ITEM);
- break;
- case KEY_RIGHT:
- menu_driver(menu, REQ_RIGHT_ITEM);
- break;
- case 10: /* ENTER */
- case 27: /* ESCAPE */
- case ' ':
- case KEY_F(F_BACK):
- case KEY_F(F_EXIT):
- break;
- }
- touchwin(win);
- refresh_all_windows(main_window);
- if (res == 10 || res == ' ') {
- res = item_index(current_item(menu));
- break;
- } else if (res == 27 || res == KEY_F(F_BACK) ||
- res == KEY_F(F_EXIT)) {
- res = KEY_EXIT;
- break;
- }
- }
- unpost_menu(menu);
- free_menu(menu);
- for (i = 0; i < btn_num; i++)
- free_item(btns[i]);
- delwin(win);
- return res;
- }
- int dialog_inputbox(WINDOW *main_window,
- const char *title, const char *prompt,
- const char *init, char **resultp, int *result_len)
- {
- int prompt_lines = 0;
- int prompt_width = 0;
- WINDOW *win;
- WINDOW *prompt_win;
- WINDOW *form_win;
- PANEL *panel;
- int i, x, y, lines, columns, win_lines, win_cols;
- int res = -1;
- int cursor_position = strlen(init);
- int cursor_form_win;
- char *result = *resultp;
- getmaxyx(stdscr, lines, columns);
- if (strlen(init)+1 > *result_len) {
- *result_len = strlen(init)+1;
- *resultp = result = xrealloc(result, *result_len);
- }
- /* find the widest line of msg: */
- prompt_lines = get_line_no(prompt);
- for (i = 0; i < prompt_lines; i++) {
- const char *line = get_line(prompt, i);
- int len = get_line_length(line);
- prompt_width = max(prompt_width, len);
- }
- if (title)
- prompt_width = max(prompt_width, strlen(title));
- win_lines = min(prompt_lines+6, lines-2);
- win_cols = min(prompt_width+7, columns-2);
- prompt_lines = max(win_lines-6, 0);
- prompt_width = max(win_cols-7, 0);
- /* place dialog in middle of screen */
- y = (lines-win_lines)/2;
- x = (columns-win_cols)/2;
- strncpy(result, init, *result_len);
- /* create the windows */
- win = newwin(win_lines, win_cols, y, x);
- prompt_win = derwin(win, prompt_lines+1, prompt_width, 2, 2);
- form_win = derwin(win, 1, prompt_width, prompt_lines+3, 2);
- keypad(form_win, TRUE);
- wattrset(form_win, attr_input_field);
- wattrset(win, attr_input_box);
- box(win, 0, 0);
- wattrset(win, attr_input_heading);
- if (title)
- mvwprintw(win, 0, 3, "%s", title);
- /* print message */
- wattrset(prompt_win, attr_input_text);
- fill_window(prompt_win, prompt);
- mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
- cursor_form_win = min(cursor_position, prompt_width-1);
- mvwprintw(form_win, 0, 0, "%s",
- result + cursor_position-cursor_form_win);
- /* create panels */
- panel = new_panel(win);
- /* show the cursor */
- curs_set(1);
- touchwin(win);
- refresh_all_windows(main_window);
- while ((res = wgetch(form_win))) {
- int len = strlen(result);
- switch (res) {
- case 10: /* ENTER */
- case 27: /* ESCAPE */
- case KEY_F(F_HELP):
- case KEY_F(F_EXIT):
- case KEY_F(F_BACK):
- break;
- case 8: /* ^H */
- case 127: /* ^? */
- case KEY_BACKSPACE:
- if (cursor_position > 0) {
- memmove(&result[cursor_position-1],
- &result[cursor_position],
- len-cursor_position+1);
- cursor_position--;
- cursor_form_win--;
- len--;
- }
- break;
- case KEY_DC:
- if (cursor_position >= 0 && cursor_position < len) {
- memmove(&result[cursor_position],
- &result[cursor_position+1],
- len-cursor_position+1);
- len--;
- }
- break;
- case KEY_UP:
- case KEY_RIGHT:
- if (cursor_position < len) {
- cursor_position++;
- cursor_form_win++;
- }
- break;
- case KEY_DOWN:
- case KEY_LEFT:
- if (cursor_position > 0) {
- cursor_position--;
- cursor_form_win--;
- }
- break;
- case KEY_HOME:
- cursor_position = 0;
- cursor_form_win = 0;
- break;
- case KEY_END:
- cursor_position = len;
- cursor_form_win = min(cursor_position, prompt_width-1);
- break;
- default:
- if ((isgraph(res) || isspace(res))) {
- /* one for new char, one for '\0' */
- if (len+2 > *result_len) {
- *result_len = len+2;
- *resultp = result = realloc(result,
- *result_len);
- }
- /* insert the char at the proper position */
- memmove(&result[cursor_position+1],
- &result[cursor_position],
- len-cursor_position+1);
- result[cursor_position] = res;
- cursor_position++;
- cursor_form_win++;
- len++;
- } else {
- mvprintw(0, 0, "unknown key: %d\n", res);
- }
- break;
- }
- if (cursor_form_win < 0)
- cursor_form_win = 0;
- else if (cursor_form_win > prompt_width-1)
- cursor_form_win = prompt_width-1;
- wmove(form_win, 0, 0);
- wclrtoeol(form_win);
- mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
- mvwprintw(form_win, 0, 0, "%s",
- result + cursor_position-cursor_form_win);
- wmove(form_win, 0, cursor_form_win);
- touchwin(win);
- refresh_all_windows(main_window);
- if (res == 10) {
- res = 0;
- break;
- } else if (res == 27 || res == KEY_F(F_BACK) ||
- res == KEY_F(F_EXIT)) {
- res = KEY_EXIT;
- break;
- } else if (res == KEY_F(F_HELP)) {
- res = 1;
- break;
- }
- }
- /* hide the cursor */
- curs_set(0);
- del_panel(panel);
- delwin(prompt_win);
- delwin(form_win);
- delwin(win);
- return res;
- }
- /* refresh all windows in the correct order */
- void refresh_all_windows(WINDOW *main_window)
- {
- update_panels();
- touchwin(main_window);
- refresh();
- }
- /* layman's scrollable window... */
- void show_scroll_win(WINDOW *main_window,
- const char *title,
- const char *text)
- {
- int res;
- int total_lines = get_line_no(text);
- int x, y, lines, columns;
- int start_x = 0, start_y = 0;
- int text_lines = 0, text_cols = 0;
- int total_cols = 0;
- int win_cols = 0;
- int win_lines = 0;
- int i = 0;
- WINDOW *win;
- WINDOW *pad;
- PANEL *panel;
- getmaxyx(stdscr, lines, columns);
- /* find the widest line of msg: */
- total_lines = get_line_no(text);
- for (i = 0; i < total_lines; i++) {
- const char *line = get_line(text, i);
- int len = get_line_length(line);
- total_cols = max(total_cols, len+2);
- }
- /* create the pad */
- pad = newpad(total_lines+10, total_cols+10);
- wattrset(pad, attr_scrollwin_text);
- fill_window(pad, text);
- win_lines = min(total_lines+4, lines-2);
- win_cols = min(total_cols+2, columns-2);
- text_lines = max(win_lines-4, 0);
- text_cols = max(win_cols-2, 0);
- /* place window in middle of screen */
- y = (lines-win_lines)/2;
- x = (columns-win_cols)/2;
- win = newwin(win_lines, win_cols, y, x);
- keypad(win, TRUE);
- /* show the help in the help window, and show the help panel */
- wattrset(win, attr_scrollwin_box);
- box(win, 0, 0);
- wattrset(win, attr_scrollwin_heading);
- mvwprintw(win, 0, 3, " %s ", title);
- panel = new_panel(win);
- /* handle scrolling */
- do {
- copywin(pad, win, start_y, start_x, 2, 2, text_lines,
- text_cols, 0);
- print_in_middle(win,
- text_lines+2,
- text_cols,
- "<OK>",
- attr_dialog_menu_fore);
- wrefresh(win);
- res = wgetch(win);
- switch (res) {
- case KEY_NPAGE:
- case ' ':
- case 'd':
- start_y += text_lines-2;
- break;
- case KEY_PPAGE:
- case 'u':
- start_y -= text_lines+2;
- break;
- case KEY_HOME:
- start_y = 0;
- break;
- case KEY_END:
- start_y = total_lines-text_lines;
- break;
- case KEY_DOWN:
- case 'j':
- start_y++;
- break;
- case KEY_UP:
- case 'k':
- start_y--;
- break;
- case KEY_LEFT:
- case 'h':
- start_x--;
- break;
- case KEY_RIGHT:
- case 'l':
- start_x++;
- break;
- }
- if (res == 10 || res == 27 || res == 'q' ||
- res == KEY_F(F_HELP) || res == KEY_F(F_BACK) ||
- res == KEY_F(F_EXIT))
- break;
- if (start_y < 0)
- start_y = 0;
- if (start_y >= total_lines-text_lines)
- start_y = total_lines-text_lines;
- if (start_x < 0)
- start_x = 0;
- if (start_x >= total_cols-text_cols)
- start_x = total_cols-text_cols;
- } while (res);
- del_panel(panel);
- delwin(win);
- refresh_all_windows(main_window);
- }
|