feat: Window click handlers

New parameters for defining fallback click handlers
that will be triggered for the whole window unless
a module action is matched.

Parameters added to all [bar/foo] sections:

- `click-left = ...`
- `click-middle= ...`
- `click-right= ...`
- `scroll-up= ...`
- `scroll-down= ...`

Ref #226
This commit is contained in:
Michael Carlberg 2016-12-05 13:17:09 +01:00
parent 3854515521
commit ec39859093
5 changed files with 81 additions and 29 deletions

View file

@ -63,6 +63,25 @@ tray-padding = 2
;override-redirect = true
; Cycle bspwm desktops when scrolling on the bar (unless caught by module)
;
; NOTE: You should probably disable scrolling for
; the bspwm module by setting `enable-scroll = false`
; in the module section
;
;scroll-up = bspc desktop -f prev.local
;scroll-down = bspc desktop -f next.local
; Cycle i3 workspaces when scrolling on the bar (unless caught by module)
;
; NOTE: You should probably disable scrolling for
; the i3 module by setting `enable-scroll = false`
; in the module section
;
;scroll-up = i3-msg workspace prev_on_output
;scroll-down = i3-msg workspace next_on_output
[module/xwindow]
type = internal/xwindow
label = %title:0:30:...%

View file

@ -73,6 +73,22 @@ struct line_settings {
uint16_t size{0U};
};
struct action {
mousebtn button{mousebtn::NONE};
string command;
};
struct action_block : public action {
alignment align{alignment::NONE};
double start_x{0.0};
double end_x{0.0};
bool active{true};
uint16_t width() const {
return static_cast<uint16_t>(end_x - start_x + 0.5);
}
};
struct bar_settings {
monitor_t monitor;
edge origin{edge::TOP};
@ -103,6 +119,8 @@ struct bar_settings {
bool override_redirect{false};
vector<action> actions;
const xcb_rectangle_t inner_area(bool abspos = false) const {
xcb_rectangle_t rect{0, 0, size.w, size.h};
@ -121,19 +139,6 @@ struct bar_settings {
}
};
struct action_block {
alignment align{alignment::NONE};
double start_x{0.0};
double end_x{0.0};
mousebtn button{mousebtn::NONE};
string command;
bool active{true};
uint16_t width() const {
return static_cast<uint16_t>(end_x - start_x + 0.5);
}
};
struct event_timer {
xcb_timestamp_t event{0L};
xcb_timestamp_t offset{1L};

View file

@ -53,10 +53,10 @@ namespace modules {
static constexpr const char* TAG_LABEL_STATE{"<label-state>"};
static constexpr const char* TAG_LABEL_MODE{"<label-mode>"};
static constexpr const char* EVENT_PREFIX{"i3"};
static constexpr const char* EVENT_CLICK{"i3-wsfocus-"};
static constexpr const char* EVENT_SCROLL_UP{"i3-wsnext"};
static constexpr const char* EVENT_SCROLL_DOWN{"i3-wsprev"};
static constexpr const char* EVENT_PREFIX{"i3wm"};
static constexpr const char* EVENT_CLICK{"i3wm-wsfocus-"};
static constexpr const char* EVENT_SCROLL_UP{"i3wm-wsnext"};
static constexpr const char* EVENT_SCROLL_DOWN{"i3wm-wsprev"};
map<state, label_t> m_statelabels;
vector<unique_ptr<workspace>> m_workspaces;

View file

@ -94,6 +94,29 @@ void bar::bootstrap(bool nodraw) {
m_opts.strut.bottom = m_conf.get<int>("global/wm", "margin-bottom", 0);
m_buttonpress.offset = xutils::event_timer_ms(m_conf, xcb_button_press_event_t{});
// Get fallback click handlers
auto click_left = m_conf.get<string>(bs, "click-left", "");
auto click_middle = m_conf.get<string>(bs, "click-middle", "");
auto click_right = m_conf.get<string>(bs, "click-right", "");
auto scroll_up = m_conf.get<string>(bs, "scroll-up", "");
auto scroll_down = m_conf.get<string>(bs, "scroll-down", "");
if (!click_left.empty()) {
m_opts.actions.emplace_back(action{mousebtn::LEFT, move(click_left)});
}
if (!click_middle.empty()) {
m_opts.actions.emplace_back(action{mousebtn::MIDDLE, move(click_middle)});
}
if (!click_right.empty()) {
m_opts.actions.emplace_back(action{mousebtn::RIGHT, move(click_right)});
}
if (!scroll_up.empty()) {
m_opts.actions.emplace_back(action{mousebtn::SCROLL_UP, move(scroll_up)});
}
if (!scroll_down.empty()) {
m_opts.actions.emplace_back(action{mousebtn::SCROLL_DOWN, move(scroll_down)});
}
}
m_log.trace("bar: Load color values");
@ -638,7 +661,6 @@ void bar::handle(const evt::button_press& evt) {
m_log.trace_x("bar: Ignoring action: end_x(%i) < event_x(%i)", action.end_x, event_x);
continue;
}
m_log.trace("Found matching input area");
m_log.trace_x("action.command = %s", action.command);
m_log.trace_x("action.button = %i", static_cast<int>(action.button));
@ -647,13 +669,23 @@ void bar::handle(const evt::button_press& evt) {
if (g_signals::bar::action_click) {
g_signals::bar::action_click(action.command);
} else {
m_log.warn("No signal handler's connected to 'action_click'");
}
return;
}
for (auto&& action : m_opts.actions) {
if (action.button == button && !action.command.empty()) {
m_log.trace("Triggering fallback click handler: %s", action.command);
if (g_signals::bar::action_click) {
g_signals::bar::action_click(action.command);
}
return;
}
}
m_log.warn("No matching input area found");
}

View file

@ -206,22 +206,18 @@ namespace modules {
}
bool i3_module::handle_event(string cmd) {
if (cmd.compare(0, 2, EVENT_PREFIX) != 0) {
return false;
}
try {
i3_util::connection_t ipc;
if (cmd.compare(0, strlen(EVENT_CLICK), EVENT_CLICK) == 0) {
m_log.info("%s: Sending workspace focus command to ipc handler", name());
ipc.send_command("workspace number " + cmd.substr(strlen(EVENT_CLICK)));
i3_util::connection_t{}.send_command("workspace number " + cmd.substr(strlen(EVENT_CLICK)));
} else if (cmd.compare(0, strlen(EVENT_SCROLL_DOWN), EVENT_SCROLL_DOWN) == 0) {
m_log.info("%s: Sending workspace prev command to ipc handler", name());
ipc.send_command("workspace next_on_output");
i3_util::connection_t{}.send_command("workspace next_on_output");
} else if (cmd.compare(0, strlen(EVENT_SCROLL_UP), EVENT_SCROLL_UP) == 0) {
m_log.info("%s: Sending workspace next command to ipc handler", name());
ipc.send_command("workspace prev_on_output");
i3_util::connection_t{}.send_command("workspace prev_on_output");
} else {
return false;
}
} catch (const exception& err) {
m_log.err("%s: %s", name(), err.what());