1 // SPDX-License-Identifier: GPL-2.0 2 #include <signal.h> 3 #include <stdbool.h> 4 #include <string.h> 5 #include <stdlib.h> 6 #include <sys/ttydefaults.h> 7 8 #include "../../util/cache.h" 9 #include "../../util/debug.h" 10 #include "../browser.h" 11 #include "../keysyms.h" 12 #include "../helpline.h" 13 #include "../ui.h" 14 #include "../util.h" 15 #include "../libslang.h" 16 17 static void ui_browser__argv_write(struct ui_browser *browser, 18 void *entry, int row) 19 { 20 char **arg = entry; 21 bool current_entry = ui_browser__is_current_entry(browser, row); 22 23 ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED : 24 HE_COLORSET_NORMAL); 25 ui_browser__write_nstring(browser, *arg, browser->width); 26 } 27 28 static int popup_menu__run(struct ui_browser *menu) 29 { 30 int key; 31 32 if (ui_browser__show(menu, " ", "ESC: exit, ENTER|->: Select option") < 0) 33 return -1; 34 35 while (1) { 36 key = ui_browser__run(menu, 0); 37 38 switch (key) { 39 case K_RIGHT: 40 case K_ENTER: 41 key = menu->index; 42 break; 43 case K_LEFT: 44 case K_ESC: 45 case 'q': 46 case CTRL('c'): 47 key = -1; 48 break; 49 default: 50 continue; 51 } 52 53 break; 54 } 55 56 ui_browser__hide(menu); 57 return key; 58 } 59 60 int ui__popup_menu(int argc, char * const argv[]) 61 { 62 struct ui_browser menu = { 63 .entries = (void *)argv, 64 .refresh = ui_browser__argv_refresh, 65 .seek = ui_browser__argv_seek, 66 .write = ui_browser__argv_write, 67 .nr_entries = argc, 68 }; 69 70 return popup_menu__run(&menu); 71 } 72 73 int ui_browser__input_window(const char *title, const char *text, char *input, 74 const char *exit_msg, int delay_secs) 75 { 76 int x, y, len, key; 77 int max_len = 60, nr_lines = 0; 78 static char buf[50]; 79 const char *t; 80 81 t = text; 82 while (1) { 83 const char *sep = strchr(t, '\n'); 84 85 if (sep == NULL) 86 sep = strchr(t, '\0'); 87 len = sep - t; 88 if (max_len < len) 89 max_len = len; 90 ++nr_lines; 91 if (*sep == '\0') 92 break; 93 t = sep + 1; 94 } 95 96 pthread_mutex_lock(&ui__lock); 97 98 max_len += 2; 99 nr_lines += 8; 100 y = SLtt_Screen_Rows / 2 - nr_lines / 2; 101 x = SLtt_Screen_Cols / 2 - max_len / 2; 102 103 SLsmg_set_color(0); 104 SLsmg_draw_box(y, x++, nr_lines, max_len); 105 if (title) { 106 SLsmg_gotorc(y, x + 1); 107 SLsmg_write_string((char *)title); 108 } 109 SLsmg_gotorc(++y, x); 110 nr_lines -= 7; 111 max_len -= 2; 112 SLsmg_write_wrapped_string((unsigned char *)text, y, x, 113 nr_lines, max_len, 1); 114 y += nr_lines; 115 len = 5; 116 while (len--) { 117 SLsmg_gotorc(y + len - 1, x); 118 SLsmg_write_nstring((char *)" ", max_len); 119 } 120 SLsmg_draw_box(y++, x + 1, 3, max_len - 2); 121 122 SLsmg_gotorc(y + 3, x); 123 SLsmg_write_nstring((char *)exit_msg, max_len); 124 SLsmg_refresh(); 125 126 pthread_mutex_unlock(&ui__lock); 127 128 x += 2; 129 len = 0; 130 key = ui__getch(delay_secs); 131 while (key != K_TIMER && key != K_ENTER && key != K_ESC) { 132 pthread_mutex_lock(&ui__lock); 133 134 if (key == K_BKSPC) { 135 if (len == 0) { 136 pthread_mutex_unlock(&ui__lock); 137 goto next_key; 138 } 139 SLsmg_gotorc(y, x + --len); 140 SLsmg_write_char(' '); 141 } else { 142 buf[len] = key; 143 SLsmg_gotorc(y, x + len++); 144 SLsmg_write_char(key); 145 } 146 SLsmg_refresh(); 147 148 pthread_mutex_unlock(&ui__lock); 149 150 /* XXX more graceful overflow handling needed */ 151 if (len == sizeof(buf) - 1) { 152 ui_helpline__push("maximum size of symbol name reached!"); 153 key = K_ENTER; 154 break; 155 } 156 next_key: 157 key = ui__getch(delay_secs); 158 } 159 160 buf[len] = '\0'; 161 strncpy(input, buf, len+1); 162 return key; 163 } 164 165 void __ui__info_window(const char *title, const char *text, const char *exit_msg) 166 { 167 int x, y; 168 int max_len = 0, nr_lines = 0; 169 const char *t; 170 171 t = text; 172 while (1) { 173 const char *sep = strchr(t, '\n'); 174 int len; 175 176 if (sep == NULL) 177 sep = strchr(t, '\0'); 178 len = sep - t; 179 if (max_len < len) 180 max_len = len; 181 ++nr_lines; 182 if (*sep == '\0') 183 break; 184 t = sep + 1; 185 } 186 187 max_len += 2; 188 nr_lines += 2; 189 if (exit_msg) 190 nr_lines += 2; 191 y = SLtt_Screen_Rows / 2 - nr_lines / 2, 192 x = SLtt_Screen_Cols / 2 - max_len / 2; 193 194 SLsmg_set_color(0); 195 SLsmg_draw_box(y, x++, nr_lines, max_len); 196 if (title) { 197 SLsmg_gotorc(y, x + 1); 198 SLsmg_write_string((char *)title); 199 } 200 SLsmg_gotorc(++y, x); 201 if (exit_msg) 202 nr_lines -= 2; 203 max_len -= 2; 204 SLsmg_write_wrapped_string((unsigned char *)text, y, x, 205 nr_lines, max_len, 1); 206 if (exit_msg) { 207 SLsmg_gotorc(y + nr_lines - 2, x); 208 SLsmg_write_nstring((char *)" ", max_len); 209 SLsmg_gotorc(y + nr_lines - 1, x); 210 SLsmg_write_nstring((char *)exit_msg, max_len); 211 } 212 } 213 214 void ui__info_window(const char *title, const char *text) 215 { 216 pthread_mutex_lock(&ui__lock); 217 __ui__info_window(title, text, NULL); 218 SLsmg_refresh(); 219 pthread_mutex_unlock(&ui__lock); 220 } 221 222 int ui__question_window(const char *title, const char *text, 223 const char *exit_msg, int delay_secs) 224 { 225 pthread_mutex_lock(&ui__lock); 226 __ui__info_window(title, text, exit_msg); 227 SLsmg_refresh(); 228 pthread_mutex_unlock(&ui__lock); 229 return ui__getch(delay_secs); 230 } 231 232 int ui__help_window(const char *text) 233 { 234 return ui__question_window("Help", text, "Press any key...", 0); 235 } 236 237 int ui__dialog_yesno(const char *msg) 238 { 239 return ui__question_window(NULL, msg, "Enter: Yes, ESC: No", 0); 240 } 241 242 static int __ui__warning(const char *title, const char *format, va_list args) 243 { 244 char *s; 245 246 if (vasprintf(&s, format, args) > 0) { 247 int key; 248 249 key = ui__question_window(title, s, "Press any key...", 0); 250 free(s); 251 return key; 252 } 253 254 fprintf(stderr, "%s\n", title); 255 vfprintf(stderr, format, args); 256 return K_ESC; 257 } 258 259 static int perf_tui__error(const char *format, va_list args) 260 { 261 return __ui__warning("Error:", format, args); 262 } 263 264 static int perf_tui__warning(const char *format, va_list args) 265 { 266 return __ui__warning("Warning:", format, args); 267 } 268 269 struct perf_error_ops perf_tui_eops = { 270 .error = perf_tui__error, 271 .warning = perf_tui__warning, 272 }; 273