blob: 5359f371d30a355c4838e3db4e170ca5e6f359c0 [file] [log] [blame]
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -03001#include "../util.h"
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -02002#include "../cache.h"
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -03003#include "../../perf.h"
Arnaldo Carvalho de Melo8f9bbc42010-08-11 14:51:47 -03004#include "libslang.h"
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -03005#include <newt.h>
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -02006#include "ui.h"
Arnaldo Carvalho de Melo8f9bbc42010-08-11 14:51:47 -03007#include <linux/compiler.h>
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -03008#include <linux/list.h>
9#include <linux/rbtree.h>
10#include <stdlib.h>
11#include <sys/ttydefaults.h>
12#include "browser.h"
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -030013#include "helpline.h"
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -020014#include "keysyms.h"
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030015#include "../color.h"
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -030016
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -020017static int ui_browser__percent_color(struct ui_browser *browser,
18 double percent, bool current)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030019{
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -020020 if (current && (!browser->use_navkeypressed || browser->navkeypressed))
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030021 return HE_COLORSET_SELECTED;
22 if (percent >= MIN_RED)
23 return HE_COLORSET_TOP;
24 if (percent >= MIN_GREEN)
25 return HE_COLORSET_MEDIUM;
26 return HE_COLORSET_NORMAL;
27}
28
Arnaldo Carvalho de Melo8f9bbc42010-08-11 14:51:47 -030029void ui_browser__set_color(struct ui_browser *self __used, int color)
30{
31 SLsmg_set_color(color);
32}
33
34void ui_browser__set_percent_color(struct ui_browser *self,
35 double percent, bool current)
36{
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -020037 int color = ui_browser__percent_color(self, percent, current);
Arnaldo Carvalho de Melo8f9bbc42010-08-11 14:51:47 -030038 ui_browser__set_color(self, color);
39}
40
41void ui_browser__gotorc(struct ui_browser *self, int y, int x)
42{
43 SLsmg_gotorc(self->y + y, self->x + x);
44}
45
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030046static struct list_head *
47ui_browser__list_head_filter_entries(struct ui_browser *browser,
48 struct list_head *pos)
49{
50 do {
51 if (!browser->filter || !browser->filter(browser, pos))
52 return pos;
53 pos = pos->next;
54 } while (pos != browser->entries);
55
56 return NULL;
57}
58
59static struct list_head *
60ui_browser__list_head_filter_prev_entries(struct ui_browser *browser,
61 struct list_head *pos)
62{
63 do {
64 if (!browser->filter || !browser->filter(browser, pos))
65 return pos;
66 pos = pos->prev;
67 } while (pos != browser->entries);
68
69 return NULL;
70}
71
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030072void ui_browser__list_head_seek(struct ui_browser *self, off_t offset, int whence)
73{
74 struct list_head *head = self->entries;
75 struct list_head *pos;
76
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030077 if (self->nr_entries == 0)
78 return;
79
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030080 switch (whence) {
81 case SEEK_SET:
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030082 pos = ui_browser__list_head_filter_entries(self, head->next);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030083 break;
84 case SEEK_CUR:
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -030085 pos = self->top;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030086 break;
87 case SEEK_END:
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030088 pos = ui_browser__list_head_filter_prev_entries(self, head->prev);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030089 break;
90 default:
91 return;
92 }
93
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030094 assert(pos != NULL);
95
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030096 if (offset > 0) {
97 while (offset-- != 0)
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -030098 pos = ui_browser__list_head_filter_entries(self, pos->next);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -030099 } else {
100 while (offset++ != 0)
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -0300101 pos = ui_browser__list_head_filter_prev_entries(self, pos->prev);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300102 }
103
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300104 self->top = pos;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300105}
106
107void ui_browser__rb_tree_seek(struct ui_browser *self, off_t offset, int whence)
108{
109 struct rb_root *root = self->entries;
110 struct rb_node *nd;
111
112 switch (whence) {
113 case SEEK_SET:
114 nd = rb_first(root);
115 break;
116 case SEEK_CUR:
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300117 nd = self->top;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300118 break;
119 case SEEK_END:
120 nd = rb_last(root);
121 break;
122 default:
123 return;
124 }
125
126 if (offset > 0) {
127 while (offset-- != 0)
128 nd = rb_next(nd);
129 } else {
130 while (offset++ != 0)
131 nd = rb_prev(nd);
132 }
133
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300134 self->top = nd;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300135}
136
137unsigned int ui_browser__rb_tree_refresh(struct ui_browser *self)
138{
139 struct rb_node *nd;
140 int row = 0;
141
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300142 if (self->top == NULL)
143 self->top = rb_first(self->entries);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300144
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300145 nd = self->top;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300146
147 while (nd != NULL) {
Arnaldo Carvalho de Melo8f9bbc42010-08-11 14:51:47 -0300148 ui_browser__gotorc(self, row, 0);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300149 self->write(self, nd, row);
150 if (++row == self->height)
151 break;
152 nd = rb_next(nd);
153 }
154
155 return row;
156}
157
158bool ui_browser__is_current_entry(struct ui_browser *self, unsigned row)
159{
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300160 return self->top_idx + row == self->index;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300161}
162
163void ui_browser__refresh_dimensions(struct ui_browser *self)
164{
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300165 self->width = SLtt_Screen_Cols - 1;
166 self->height = SLtt_Screen_Rows - 2;
Arnaldo Carvalho de Melo469917c2010-09-13 10:25:04 -0300167 self->y = 1;
168 self->x = 0;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300169}
170
171void ui_browser__reset_index(struct ui_browser *self)
172{
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300173 self->index = self->top_idx = 0;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300174 self->seek(self, 0, SEEK_SET);
175}
176
Arnaldo Carvalho de Melob210b3b2011-02-25 11:33:31 -0300177void __ui_browser__show_title(struct ui_browser *browser, const char *title)
178{
179 SLsmg_gotorc(0, 0);
180 ui_browser__set_color(browser, NEWT_COLORSET_ROOT);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300181 slsmg_write_nstring(title, browser->width + 1);
Arnaldo Carvalho de Melob210b3b2011-02-25 11:33:31 -0300182}
183
184void ui_browser__show_title(struct ui_browser *browser, const char *title)
185{
186 pthread_mutex_lock(&ui__lock);
187 __ui_browser__show_title(browser, title);
188 pthread_mutex_unlock(&ui__lock);
189}
190
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300191int ui_browser__show(struct ui_browser *self, const char *title,
192 const char *helpline, ...)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300193{
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300194 int err;
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300195 va_list ap;
196
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300197 ui_browser__refresh_dimensions(self);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300198
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200199 pthread_mutex_lock(&ui__lock);
Arnaldo Carvalho de Melob210b3b2011-02-25 11:33:31 -0300200 __ui_browser__show_title(self, title);
Arnaldo Carvalho de Melo469917c2010-09-13 10:25:04 -0300201
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300202 self->title = title;
203 free(self->helpline);
204 self->helpline = NULL;
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300205
206 va_start(ap, helpline);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300207 err = vasprintf(&self->helpline, helpline, ap);
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300208 va_end(ap);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300209 if (err > 0)
210 ui_helpline__push(self->helpline);
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200211 pthread_mutex_unlock(&ui__lock);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300212 return err ? 0 : -1;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300213}
214
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300215void ui_browser__hide(struct ui_browser *browser __used)
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300216{
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200217 pthread_mutex_lock(&ui__lock);
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300218 ui_helpline__pop();
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200219 pthread_mutex_unlock(&ui__lock);
Arnaldo Carvalho de Melo59e8fe32010-08-10 15:44:20 -0300220}
221
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300222static void ui_browser__scrollbar_set(struct ui_browser *browser)
223{
224 int height = browser->height, h = 0, pct = 0,
225 col = browser->width,
226 row = browser->y - 1;
227
228 if (browser->nr_entries > 1) {
229 pct = ((browser->index * (browser->height - 1)) /
230 (browser->nr_entries - 1));
231 }
232
233 while (h < height) {
234 ui_browser__gotorc(browser, row++, col);
235 SLsmg_set_char_set(1);
236 SLsmg_write_char(h == pct ? SLSMG_DIAMOND_CHAR : SLSMG_BOARD_CHAR);
237 SLsmg_set_char_set(0);
238 ++h;
239 }
240}
241
242static int __ui_browser__refresh(struct ui_browser *browser)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300243{
244 int row;
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -0200245 int width = browser->width;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300246
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300247 row = browser->refresh(browser);
248 ui_browser__set_color(browser, HE_COLORSET_NORMAL);
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -0200249
250 if (!browser->use_navkeypressed || browser->navkeypressed)
251 ui_browser__scrollbar_set(browser);
252 else
253 width += 1;
254
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300255 SLsmg_fill_region(browser->y + row, browser->x,
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -0200256 browser->height - row, width, ' ');
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300257
258 return 0;
259}
260
261int ui_browser__refresh(struct ui_browser *browser)
262{
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200263 pthread_mutex_lock(&ui__lock);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300264 __ui_browser__refresh(browser);
Arnaldo Carvalho de Melo5c35d692011-02-09 11:38:43 -0200265 pthread_mutex_unlock(&ui__lock);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300266
267 return 0;
268}
269
Arnaldo Carvalho de Melo900e14a2011-10-11 16:15:39 -0300270/*
271 * Here we're updating nr_entries _after_ we started browsing, i.e. we have to
272 * forget about any reference to any entry in the underlying data structure,
273 * that is why we do a SEEK_SET. Think about 'perf top' in the hists browser
274 * after an output_resort and hist decay.
275 */
276void ui_browser__update_nr_entries(struct ui_browser *browser, u32 nr_entries)
277{
278 off_t offset = nr_entries - browser->nr_entries;
279
280 browser->nr_entries = nr_entries;
281
282 if (offset < 0) {
283 if (browser->top_idx < (u64)-offset)
284 offset = -browser->top_idx;
285
286 browser->index += offset;
287 browser->top_idx += offset;
288 }
289
Arnaldo Carvalho de Melo437cfe72011-10-14 09:31:53 -0300290 browser->top = NULL;
Arnaldo Carvalho de Melo900e14a2011-10-11 16:15:39 -0300291 browser->seek(browser, browser->top_idx, SEEK_SET);
292}
293
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200294static int ui__getch(int delay_secs)
295{
296 struct timeval timeout, *ptimeout = delay_secs ? &timeout : NULL;
297 fd_set read_set;
298 int err, key;
299
300 FD_ZERO(&read_set);
301 FD_SET(0, &read_set);
302
303 if (delay_secs) {
304 timeout.tv_sec = delay_secs;
305 timeout.tv_usec = 0;
306 }
307
308 err = select(1, &read_set, NULL, NULL, ptimeout);
309
310 if (err == 0)
311 return K_TIMER;
312
313 if (err == -1) {
314 if (errno == EINTR)
315 return K_RESIZE;
316 return K_ERROR;
317 }
318
319 key = SLang_getkey();
320 if (key != K_ESC)
321 return key;
322
323 FD_ZERO(&read_set);
324 FD_SET(0, &read_set);
325 timeout.tv_sec = 0;
326 timeout.tv_usec = 20;
327 err = select(1, &read_set, NULL, NULL, &timeout);
328 if (err == 0)
329 return K_ESC;
330
331 SLang_ungetkey(key);
332 return SLkp_getkey();
333}
334
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300335int ui_browser__run(struct ui_browser *self, int delay_secs)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300336{
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300337 int err, key;
Arnaldo Carvalho de Melob50e0032010-08-11 10:07:43 -0300338
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300339 pthread__unblock_sigwinch();
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300340
341 while (1) {
342 off_t offset;
343
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300344 pthread_mutex_lock(&ui__lock);
345 err = __ui_browser__refresh(self);
346 SLsmg_refresh();
347 pthread_mutex_unlock(&ui__lock);
348 if (err < 0)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300349 break;
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300350
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200351 key = ui__getch(delay_secs);
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300352
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200353 if (key == K_RESIZE) {
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300354 pthread_mutex_lock(&ui__lock);
355 SLtt_get_screen_size();
356 SLsmg_reinit_smg();
357 pthread_mutex_unlock(&ui__lock);
358 ui_browser__refresh_dimensions(self);
359 __ui_browser__show_title(self, self->title);
360 ui_helpline__puts(self->helpline);
361 continue;
362 }
363
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -0200364 if (self->use_navkeypressed && !self->navkeypressed) {
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200365 if (key == K_DOWN || key == K_UP ||
366 key == K_PGDN || key == K_PGUP ||
367 key == K_HOME || key == K_END ||
Arnaldo Carvalho de Meloc172f742011-10-18 14:31:35 -0200368 key == ' ') {
369 self->navkeypressed = true;
370 continue;
371 } else
372 return key;
373 }
374
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300375 switch (key) {
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200376 case K_DOWN:
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300377 if (self->index == self->nr_entries - 1)
378 break;
379 ++self->index;
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300380 if (self->index == self->top_idx + self->height) {
381 ++self->top_idx;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300382 self->seek(self, +1, SEEK_CUR);
383 }
384 break;
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200385 case K_UP:
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300386 if (self->index == 0)
387 break;
388 --self->index;
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300389 if (self->index < self->top_idx) {
390 --self->top_idx;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300391 self->seek(self, -1, SEEK_CUR);
392 }
393 break;
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200394 case K_PGDN:
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300395 case ' ':
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300396 if (self->top_idx + self->height > self->nr_entries - 1)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300397 break;
398
399 offset = self->height;
400 if (self->index + offset > self->nr_entries - 1)
401 offset = self->nr_entries - 1 - self->index;
402 self->index += offset;
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300403 self->top_idx += offset;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300404 self->seek(self, +offset, SEEK_CUR);
405 break;
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200406 case K_PGUP:
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300407 if (self->top_idx == 0)
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300408 break;
409
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300410 if (self->top_idx < self->height)
411 offset = self->top_idx;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300412 else
413 offset = self->height;
414
415 self->index -= offset;
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300416 self->top_idx -= offset;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300417 self->seek(self, -offset, SEEK_CUR);
418 break;
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200419 case K_HOME:
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300420 ui_browser__reset_index(self);
421 break;
Arnaldo Carvalho de Melocf958002011-10-20 16:59:15 -0200422 case K_END:
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300423 offset = self->height - 1;
424 if (offset >= self->nr_entries)
425 offset = self->nr_entries - 1;
426
427 self->index = self->nr_entries - 1;
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300428 self->top_idx = self->index - offset;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300429 self->seek(self, -offset, SEEK_END);
430 break;
431 default:
Arnaldo Carvalho de Melo3af6e332011-10-13 08:52:46 -0300432 return key;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300433 }
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300434 }
Arnaldo Carvalho de Melob50e0032010-08-11 10:07:43 -0300435 return -1;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300436}
437
438unsigned int ui_browser__list_head_refresh(struct ui_browser *self)
439{
440 struct list_head *pos;
441 struct list_head *head = self->entries;
442 int row = 0;
443
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300444 if (self->top == NULL || self->top == self->entries)
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -0300445 self->top = ui_browser__list_head_filter_entries(self, head->next);
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300446
Arnaldo Carvalho de Melod247eb62010-08-07 13:56:04 -0300447 pos = self->top;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300448
449 list_for_each_from(pos, head) {
Arnaldo Carvalho de Melo250611c2011-10-14 12:27:54 -0300450 if (!self->filter || !self->filter(self, pos)) {
451 ui_browser__gotorc(self, row, 0);
452 self->write(self, pos, row);
453 if (++row == self->height)
454 break;
455 }
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300456 }
457
458 return row;
459}
460
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200461static struct ui_browser__colorset {
462 const char *name, *fg, *bg;
463 int colorset;
464} ui_browser__colorsets[] = {
465 {
466 .colorset = HE_COLORSET_TOP,
467 .name = "top",
468 .fg = "red",
Arnaldo Carvalho de Melo82e0af82011-10-19 00:30:32 -0200469 .bg = "default",
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200470 },
471 {
472 .colorset = HE_COLORSET_MEDIUM,
473 .name = "medium",
474 .fg = "green",
Arnaldo Carvalho de Melo82e0af82011-10-19 00:30:32 -0200475 .bg = "default",
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200476 },
477 {
478 .colorset = HE_COLORSET_NORMAL,
479 .name = "normal",
Arnaldo Carvalho de Melo82e0af82011-10-19 00:30:32 -0200480 .fg = "default",
481 .bg = "default",
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200482 },
483 {
484 .colorset = HE_COLORSET_SELECTED,
485 .name = "selected",
486 .fg = "black",
487 .bg = "lightgray",
488 },
489 {
490 .colorset = HE_COLORSET_CODE,
491 .name = "code",
492 .fg = "blue",
Arnaldo Carvalho de Melo82e0af82011-10-19 00:30:32 -0200493 .bg = "default",
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200494 },
495 {
496 .name = NULL,
497 }
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300498};
499
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200500
501static int ui_browser__color_config(const char *var, const char *value,
502 void *data __used)
503{
504 char *fg = NULL, *bg;
505 int i;
506
507 /* same dir for all commands */
508 if (prefixcmp(var, "colors.") != 0)
509 return 0;
510
511 for (i = 0; ui_browser__colorsets[i].name != NULL; ++i) {
512 const char *name = var + 7;
513
514 if (strcmp(ui_browser__colorsets[i].name, name) != 0)
515 continue;
516
517 fg = strdup(value);
518 if (fg == NULL)
519 break;
520
521 bg = strchr(fg, ',');
522 if (bg == NULL)
523 break;
524
525 *bg = '\0';
526 while (isspace(*++bg));
527 ui_browser__colorsets[i].bg = bg;
528 ui_browser__colorsets[i].fg = fg;
529 return 0;
530 }
531
532 free(fg);
533 return -1;
534}
535
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300536void ui_browser__init(void)
537{
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200538 int i = 0;
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300539
Arnaldo Carvalho de Meloe039fc72011-10-18 15:50:51 -0200540 perf_config(ui_browser__color_config, NULL);
541
542 while (ui_browser__colorsets[i].name) {
543 struct ui_browser__colorset *c = &ui_browser__colorsets[i++];
544 sltt_set_color(c->colorset, c->name, c->fg, c->bg);
545 }
Arnaldo Carvalho de Meloef8f34a2010-08-06 17:35:02 -0300546}