From a6a5f736d4f4aecb103761a8655d9b17fb74368a Mon Sep 17 00:00:00 2001 From: rootcoma <rootcoma@gmail.com> Date: Thu, 7 Aug 2014 01:08:54 -0700 Subject: [PATCH] Add feature multiple actions #66 --- bar.c | 101 +++++++++++++++++++++++++++++++++++++++++++++------------- 1 file changed, 78 insertions(+), 23 deletions(-) diff --git a/bar.c b/bar.c index 3fed67e..820c9d3 100644 --- a/bar.c +++ b/bar.c @@ -46,6 +46,11 @@ typedef struct area_stack_t { area_t slot[N]; } area_stack_t; +typedef struct area_queue_t { + int pos; + struct area_queue_t *next; +} area_queue_t; + enum { ATTR_OVERL = (1<<0), ATTR_UNDERL = (1<<1), @@ -85,6 +90,7 @@ static int bu = 1; /* Underline height */ static uint32_t fgc, bgc, ugc; static uint32_t dfgc, dbgc; static area_stack_t astack; +static area_queue_t *aqueuehead; void update_gc (void) @@ -189,7 +195,6 @@ parse_color (const char *str, char **end, const uint32_t def) return ret; } - void set_attribute (const char modifier, const char attribute) { @@ -207,14 +212,53 @@ set_attribute (const char modifier, const char attribute) } } - -area_t * -area_get (xcb_window_t win, const int x) +void +area_handle_click (xcb_window_t win, const int x, const int button) { for (int i = 0; i < astack.pos; i++) - if (astack.slot[i].window == win && x > astack.slot[i].begin && x < astack.slot[i].end) - return &astack.slot[i]; - return NULL; + if (astack.slot[i].window == win && x > astack.slot[i].begin && x < astack.slot[i].end + && button == astack.slot[i].button) { + write(STDOUT_FILENO, astack.slot[i].cmd, strlen(astack.slot[i].cmd)); + write(STDOUT_FILENO, "\n", 1); + } +} + +int +area_queue_pop () +{ + int pos = -1; + if (!aqueuehead) + return pos; + + area_queue_t *a = aqueuehead->next; + pos = aqueuehead->pos; + free(aqueuehead); + aqueuehead = a; + return pos; +} + +void +area_queue_free () +{ + while (aqueuehead) { + area_queue_t *a = aqueuehead->next; + free(aqueuehead); + aqueuehead = a; + } +} + +void +area_queue_add (const int pos) +{ + area_queue_t *ret; + ret = calloc(1, sizeof(area_queue_t)); + if (!ret) { + fprintf(stderr, "Failed to allocate new area queue\n"); + exit(EXIT_FAILURE); + } + ret->next = aqueuehead; + ret->pos = pos; + aqueuehead = ret; } void @@ -227,8 +271,11 @@ area_shift (xcb_window_t win, const int align, int delta) for (int i = 0; i < astack.pos; i++) { if (astack.slot[i].window == win && astack.slot[i].align == align) { - astack.slot[i].begin -= delta; - astack.slot[i].end -= delta; + /* make sure area tag is closed before moving the bounds */ + if (astack.slot[i].end != 0) { + astack.slot[i].begin -= delta; + astack.slot[i].end -= delta; + } } } } @@ -237,7 +284,6 @@ bool area_add (char *str, const char *optend, char **end, monitor_t *mon, const int x, const int align, const int button) { char *p = str; - area_t *a = &astack.slot[astack.pos]; if (astack.pos == N) { fprintf(stderr, "astack overflow!\n"); @@ -248,8 +294,14 @@ area_add (char *str, const char *optend, char **end, monitor_t *mon, const int x if (*p != ':') { *end = p; + const int pos = area_queue_pop(); + if (pos == -1) + return false; + + area_t *a = &astack.slot[pos]; + /* Basic safety checks */ - if (!a->cmd || a->align != align || a->window != mon->window) + if (!a || !a->cmd || a->align != align || a->window != mon->window) return false; const int size = x - a->begin; @@ -269,8 +321,6 @@ area_add (char *str, const char *optend, char **end, monitor_t *mon, const int x break; } - astack.pos++; - return true; } @@ -284,6 +334,8 @@ area_add (char *str, const char *optend, char **end, monitor_t *mon, const int x *trail = '\0'; + area_t *a = &astack.slot[astack.pos]; + /* This is a pointer to the string buffer allocated in the main */ a->cmd = p; a->align = align; @@ -292,7 +344,8 @@ area_add (char *str, const char *optend, char **end, monitor_t *mon, const int x a->button = button; *end = trail + 1; - + area_queue_add(astack.pos); + astack.pos++; return true; } @@ -335,6 +388,7 @@ parse (char *text) align = ALIGN_L; cur_mon = monhead; + area_queue_free(); memset(&astack, 0, sizeof(area_stack_t)); for (monitor_t *m = monhead; m != NULL; m = m->next) @@ -742,11 +796,11 @@ get_randr_monitors (void) return; } - xcb_rectangle_t r[valid]; + xcb_rectangle_t r[valid]; - for (i = j = 0; i < num && j < valid; i++) - if (rects[i].width != 0) - r[j++] = rects[i]; + for (i = j = 0; i < num && j < valid; i++) + if (rects[i].width != 0) + r[j++] = rects[i]; monitor_create_chain(r, valid); } @@ -855,6 +909,9 @@ init (void) /* Initialiaze monitor list head and tail */ monhead = montail = NULL; + /* Initialize area queue list head */ + aqueuehead = NULL; + /* Check if RandR is present */ qe_reply = xcb_get_extension_data(c, &xcb_randr_id); @@ -922,6 +979,8 @@ init (void) void cleanup (void) { + area_queue_free(); + for (int i = 0; font_list[i]; i++) { xcb_close_font(c, font_list[i]->ptr); free(font_list[i]); @@ -1119,12 +1178,8 @@ main (int argc, char **argv) case XCB_BUTTON_PRESS: press_ev = (xcb_button_press_event_t *)ev; { - area_t *area = area_get(press_ev->event, press_ev->event_x); /* Respond to the click */ - if (area && area->button == press_ev->detail) { - write(STDOUT_FILENO, area->cmd, strlen(area->cmd)); - write(STDOUT_FILENO, "\n", 1); - } + area_handle_click(press_ev->event, press_ev->event_x, press_ev->detail); } break; }