Merge tag 'block-5.9-2020-08-14' of git://git.kernel.dk/linux-block
[linux/fpc-iii.git] / tools / perf / ui / tui / util.c
blob0f562e2cb1e881181976ad1e1bbad77aa939c580
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>
8 #include "../browser.h"
9 #include "../keysyms.h"
10 #include "../helpline.h"
11 #include "../ui.h"
12 #include "../util.h"
13 #include "../libslang.h"
15 static void ui_browser__argv_write(struct ui_browser *browser,
16 void *entry, int row)
18 char **arg = entry;
19 bool current_entry = ui_browser__is_current_entry(browser, row);
21 ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED :
22 HE_COLORSET_NORMAL);
23 ui_browser__write_nstring(browser, *arg, browser->width);
26 static int popup_menu__run(struct ui_browser *menu, int *keyp)
28 int key;
30 if (ui_browser__show(menu, " ", "ESC: exit, ENTER|->: Select option") < 0)
31 return -1;
33 while (1) {
34 key = ui_browser__run(menu, 0);
36 switch (key) {
37 case K_RIGHT:
38 case K_ENTER:
39 key = menu->index;
40 break;
41 case K_LEFT:
42 case K_ESC:
43 case 'q':
44 case CTRL('c'):
45 key = -1;
46 break;
47 default:
48 if (keyp) {
49 *keyp = key;
50 key = menu->nr_entries;
51 break;
53 continue;
56 break;
59 ui_browser__hide(menu);
60 return key;
63 int ui__popup_menu(int argc, char * const argv[], int *keyp)
65 struct ui_browser menu = {
66 .entries = (void *)argv,
67 .refresh = ui_browser__argv_refresh,
68 .seek = ui_browser__argv_seek,
69 .write = ui_browser__argv_write,
70 .nr_entries = argc,
72 return popup_menu__run(&menu, keyp);
75 int ui_browser__input_window(const char *title, const char *text, char *input,
76 const char *exit_msg, int delay_secs)
78 int x, y, len, key;
79 int max_len = 60, nr_lines = 0;
80 static char buf[50];
81 const char *t;
83 t = text;
84 while (1) {
85 const char *sep = strchr(t, '\n');
87 if (sep == NULL)
88 sep = strchr(t, '\0');
89 len = sep - t;
90 if (max_len < len)
91 max_len = len;
92 ++nr_lines;
93 if (*sep == '\0')
94 break;
95 t = sep + 1;
98 pthread_mutex_lock(&ui__lock);
100 max_len += 2;
101 nr_lines += 8;
102 y = SLtt_Screen_Rows / 2 - nr_lines / 2;
103 x = SLtt_Screen_Cols / 2 - max_len / 2;
105 SLsmg_set_color(0);
106 SLsmg_draw_box(y, x++, nr_lines, max_len);
107 if (title) {
108 SLsmg_gotorc(y, x + 1);
109 SLsmg_write_string((char *)title);
111 SLsmg_gotorc(++y, x);
112 nr_lines -= 7;
113 max_len -= 2;
114 SLsmg_write_wrapped_string((unsigned char *)text, y, x,
115 nr_lines, max_len, 1);
116 y += nr_lines;
117 len = 5;
118 while (len--) {
119 SLsmg_gotorc(y + len - 1, x);
120 SLsmg_write_nstring((char *)" ", max_len);
122 SLsmg_draw_box(y++, x + 1, 3, max_len - 2);
124 SLsmg_gotorc(y + 3, x);
125 SLsmg_write_nstring((char *)exit_msg, max_len);
126 SLsmg_refresh();
128 pthread_mutex_unlock(&ui__lock);
130 x += 2;
131 len = 0;
132 key = ui__getch(delay_secs);
133 while (key != K_TIMER && key != K_ENTER && key != K_ESC) {
134 pthread_mutex_lock(&ui__lock);
136 if (key == K_BKSPC) {
137 if (len == 0) {
138 pthread_mutex_unlock(&ui__lock);
139 goto next_key;
141 SLsmg_gotorc(y, x + --len);
142 SLsmg_write_char(' ');
143 } else {
144 buf[len] = key;
145 SLsmg_gotorc(y, x + len++);
146 SLsmg_write_char(key);
148 SLsmg_refresh();
150 pthread_mutex_unlock(&ui__lock);
152 /* XXX more graceful overflow handling needed */
153 if (len == sizeof(buf) - 1) {
154 ui_helpline__push("maximum size of symbol name reached!");
155 key = K_ENTER;
156 break;
158 next_key:
159 key = ui__getch(delay_secs);
162 buf[len] = '\0';
163 strncpy(input, buf, len+1);
164 return key;
167 void __ui__info_window(const char *title, const char *text, const char *exit_msg)
169 int x, y;
170 int max_len = 0, nr_lines = 0;
171 const char *t;
173 t = text;
174 while (1) {
175 const char *sep = strchr(t, '\n');
176 int len;
178 if (sep == NULL)
179 sep = strchr(t, '\0');
180 len = sep - t;
181 if (max_len < len)
182 max_len = len;
183 ++nr_lines;
184 if (*sep == '\0')
185 break;
186 t = sep + 1;
189 max_len += 2;
190 nr_lines += 2;
191 if (exit_msg)
192 nr_lines += 2;
193 y = SLtt_Screen_Rows / 2 - nr_lines / 2,
194 x = SLtt_Screen_Cols / 2 - max_len / 2;
196 SLsmg_set_color(0);
197 SLsmg_draw_box(y, x++, nr_lines, max_len);
198 if (title) {
199 SLsmg_gotorc(y, x + 1);
200 SLsmg_write_string((char *)title);
202 SLsmg_gotorc(++y, x);
203 if (exit_msg)
204 nr_lines -= 2;
205 max_len -= 2;
206 SLsmg_write_wrapped_string((unsigned char *)text, y, x,
207 nr_lines, max_len, 1);
208 if (exit_msg) {
209 SLsmg_gotorc(y + nr_lines - 2, x);
210 SLsmg_write_nstring((char *)" ", max_len);
211 SLsmg_gotorc(y + nr_lines - 1, x);
212 SLsmg_write_nstring((char *)exit_msg, max_len);
216 void ui__info_window(const char *title, const char *text)
218 pthread_mutex_lock(&ui__lock);
219 __ui__info_window(title, text, NULL);
220 SLsmg_refresh();
221 pthread_mutex_unlock(&ui__lock);
224 int ui__question_window(const char *title, const char *text,
225 const char *exit_msg, int delay_secs)
227 pthread_mutex_lock(&ui__lock);
228 __ui__info_window(title, text, exit_msg);
229 SLsmg_refresh();
230 pthread_mutex_unlock(&ui__lock);
231 return ui__getch(delay_secs);
234 int ui__help_window(const char *text)
236 return ui__question_window("Help", text, "Press any key...", 0);
239 int ui__dialog_yesno(const char *msg)
241 return ui__question_window(NULL, msg, "Enter: Yes, ESC: No", 0);
244 static int __ui__warning(const char *title, const char *format, va_list args)
246 char *s;
248 if (vasprintf(&s, format, args) > 0) {
249 int key;
251 key = ui__question_window(title, s, "Press any key...", 0);
252 free(s);
253 return key;
256 fprintf(stderr, "%s\n", title);
257 vfprintf(stderr, format, args);
258 return K_ESC;
261 static int perf_tui__error(const char *format, va_list args)
263 return __ui__warning("Error:", format, args);
266 static int perf_tui__warning(const char *format, va_list args)
268 return __ui__warning("Warning:", format, args);
271 struct perf_error_ops perf_tui_eops = {
272 .error = perf_tui__error,
273 .warning = perf_tui__warning,