handle_button: Introduce child_handle_button
Also fixes an issue where action would be called if the button press was on a separator. For example, if a user scrolled on a separator, the workspace would change. Applies to --release commands as well.
This commit is contained in:
parent
03d2ccdeef
commit
c2a1b6e91f
100
i3bar/src/xcb.c
100
i3bar/src/xcb.c
|
@ -454,6 +454,61 @@ static bool execute_custom_command(xcb_keycode_t input_code, bool event_is_relea
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void child_handle_button(xcb_button_press_event_t *event, i3_output *output, int32_t x) {
|
||||||
|
if (!child_want_click_events()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const int tray_width = get_tray_width(output->trayclients);
|
||||||
|
/* Calculate the horizontal coordinate (x) of the start of the statusline by
|
||||||
|
* subtracting its width and the width of the tray from the bar width. */
|
||||||
|
const int offset = output->rect.w - output->statusline_width - tray_width - logical_px((tray_width > 0) * sb_hoff_px);
|
||||||
|
/* x of the click event relative to the start of the statusline. */
|
||||||
|
const uint32_t statusline_x = x - offset;
|
||||||
|
|
||||||
|
if (x < offset || statusline_x > (uint32_t)output->statusline_width) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* x of the start of the current block relative to the statusline. */
|
||||||
|
uint32_t last_block_x = 0;
|
||||||
|
struct status_block *block;
|
||||||
|
TAILQ_FOREACH(block, &statusline_head, blocks) {
|
||||||
|
i3String *text;
|
||||||
|
struct status_block_render_desc *render;
|
||||||
|
if (output->statusline_short_text && block->short_text != NULL) {
|
||||||
|
text = block->short_text;
|
||||||
|
render = &block->short_render;
|
||||||
|
} else {
|
||||||
|
text = block->full_text;
|
||||||
|
render = &block->full_render;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i3string_get_num_bytes(text) == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Include the whole block in our calculations: when min_width is
|
||||||
|
* specified, we have to take padding width into account. */
|
||||||
|
const uint32_t full_render_width = render->width + render->x_offset + render->x_append;
|
||||||
|
/* x of the click event relative to the current block. */
|
||||||
|
const uint32_t relative_x = statusline_x - last_block_x;
|
||||||
|
if (relative_x <= full_render_width) {
|
||||||
|
send_block_clicked(event->detail, block->name, block->instance,
|
||||||
|
event->root_x, event->root_y, relative_x,
|
||||||
|
event->event_y, full_render_width, bar_height,
|
||||||
|
event->state);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
last_block_x += full_render_width + block->sep_block_width;
|
||||||
|
if (last_block_x > statusline_x) {
|
||||||
|
/* Click was on a separator. */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Handle a button press event (i.e. a mouse click on one of our bars).
|
* Handle a button press event (i.e. a mouse click on one of our bars).
|
||||||
* We determine, whether the click occurred on a workspace button or if the scroll-
|
* We determine, whether the click occurred on a workspace button or if the scroll-
|
||||||
|
@ -479,10 +534,6 @@ static void handle_button(xcb_button_press_event_t *event) {
|
||||||
|
|
||||||
/* During button release events, only check for custom commands. */
|
/* During button release events, only check for custom commands. */
|
||||||
const bool event_is_release = (event->response_type & ~0x80) == XCB_BUTTON_RELEASE;
|
const bool event_is_release = (event->response_type & ~0x80) == XCB_BUTTON_RELEASE;
|
||||||
if (event_is_release) {
|
|
||||||
execute_custom_command(event->detail, event_is_release);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int32_t x = event->event_x >= 0 ? event->event_x : 0;
|
int32_t x = event->event_x >= 0 ? event->event_x : 0;
|
||||||
int workspace_width = 0;
|
int workspace_width = 0;
|
||||||
|
@ -499,45 +550,18 @@ static void handle_button(xcb_button_press_event_t *event) {
|
||||||
workspace_width += logical_px(ws_spacing_px);
|
workspace_width += logical_px(ws_spacing_px);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (x > workspace_width && child_want_click_events()) {
|
if (x > workspace_width) {
|
||||||
/* If the child asked for click events,
|
if (!event_is_release) {
|
||||||
* check if a status block has been clicked. */
|
child_handle_button(event, walk, x);
|
||||||
int tray_width = get_tray_width(walk->trayclients);
|
|
||||||
int last_block_x = 0;
|
|
||||||
int offset = walk->rect.w - walk->statusline_width - tray_width - logical_px((tray_width > 0) * sb_hoff_px);
|
|
||||||
int32_t statusline_x = x - offset;
|
|
||||||
|
|
||||||
if (statusline_x >= 0 && statusline_x < walk->statusline_width) {
|
|
||||||
struct status_block *block;
|
|
||||||
|
|
||||||
TAILQ_FOREACH(block, &statusline_head, blocks) {
|
|
||||||
i3String *text = block->full_text;
|
|
||||||
struct status_block_render_desc *render = &block->full_render;
|
|
||||||
if (walk->statusline_short_text && block->short_text != NULL) {
|
|
||||||
text = block->short_text;
|
|
||||||
render = &block->short_render;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (i3string_get_num_bytes(text) == 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
const uint32_t full_render_width = render->width + render->x_offset + render->x_append;
|
|
||||||
const int relative_x = statusline_x - last_block_x;
|
|
||||||
if (relative_x >= 0 && (uint32_t)relative_x <= full_render_width) {
|
|
||||||
send_block_clicked(event->detail, block->name, block->instance,
|
|
||||||
event->root_x, event->root_y, relative_x, event->event_y, full_render_width, bar_height,
|
|
||||||
event->state);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
last_block_x += full_render_width + block->sep_block_width;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
/* Return to avoid executing any other actions when a separator is
|
||||||
|
* clicked. */
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If a custom command was specified for this mouse button, it overrides
|
/* If a custom command was specified for this mouse button, it overrides
|
||||||
* the default behavior. */
|
* the default behavior. */
|
||||||
if (execute_custom_command(event->detail, event_is_release)) {
|
if (execute_custom_command(event->detail, event_is_release) || event_is_release) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue