#pragma once #include #include #include "common.hpp" #include "components/x11/connection.hpp" #include "components/x11/randr.hpp" #include "components/x11/window.hpp" #include "config.hpp" #include "utils/socket.hpp" #include "utils/string.hpp" LEMONBUDDY_NS namespace bspwm_util { struct payload; using connection_t = unique_ptr; using payload_t = unique_ptr; /** * bspwm payload */ struct payload { char data[BUFSIZ]{'\0'}; size_t len = 0; }; /** * Get all bspwm root windows */ auto root_windows(connection& conn) { vector roots; auto children = conn.query_tree(conn.screen()->root).children(); for (auto it = children.begin(); it != children.end(); it++) { auto cookie = xcb_icccm_get_wm_class(conn, *it); xcb_icccm_get_wm_class_reply_t reply; if (xcb_icccm_get_wm_class_reply(conn, cookie, &reply, nullptr) == 0) continue; if (!string_util::compare("Bspwm", reply.class_name) || !string_util::compare("root", reply.instance_name)) continue; roots.emplace_back(*it); } return roots; } /** * Restack given window above the bspwm root window * for the given monitor. * * Fixes the issue with always-on-top window's */ bool restack_above_root(connection& conn, const monitor_t& mon, const xcb_window_t win) { for (auto&& root : root_windows(conn)) { auto geom = conn.get_geometry(root); if (mon->x != geom->x || mon->y != geom->y) continue; if (mon->w != geom->width || mon->h != geom->height) continue; const uint32_t value_mask = XCB_CONFIG_WINDOW_SIBLING | XCB_CONFIG_WINDOW_STACK_MODE; const uint32_t value_list[2]{root, XCB_STACK_MODE_ABOVE}; conn.configure_window_checked(win, value_mask, value_list); conn.flush(); return true; } return false; } /** * Get path to the bspwm socket by the following order * * 1. Value of environment variable BSPWM_SOCKET * 2. Value built from the bspwm socket path template * 3. Value of the macro BSPWM_SOCKET_PATH */ string get_socket_path() { string env_path; if ((env_path = read_env("BSPWM_SOCKET")).empty() == false) return env_path; struct sockaddr_un sa; char* host = nullptr; int dsp = 0; int scr = 0; if (xcb_parse_display(nullptr, &host, &dsp, &scr) == 0) return BSPWM_SOCKET_PATH; snprintf(sa.sun_path, sizeof(sa.sun_path), "/tmp/bspwm%s_%i_%i-socket", host, dsp, scr); return sa.sun_path; } /** * Generate a payload object with properly formatted data * ready to be sent to the bspwm ipc controller */ unique_ptr make_payload(string cmd) { auto pl = make_unique(); auto size = sizeof(pl->data); int offset = 0; int chars = 0; for (auto&& word : string_util::split(cmd, ' ')) { chars = snprintf(pl->data + offset, size - offset, "%s%c", word.c_str(), 0); pl->len += chars; offset += chars; } return pl; } /** * Create an ipc socket connection * * Example usage: * @code cpp * auto ipc = bspwm_util::make_connection(); * ipc->send(bspwm_util::make_payload("desktop -f eDP-1:^1")); * @endcode */ connection_t make_connection() { return socket_util::make_unix_connection(get_socket_path()); } /** * Create a connection and subscribe to events * on the bspwm socket * * Example usage: * @code cpp * auto ipc = bspwm_util::make_subscriber(); * * while (!ipc->poll(POLLHUP, 0)) { * ssize_t bytes_received = 0; * auto data = ipc->receive(BUFSIZ-1, bytes_received, 0); * std::cout << data << std::endl; * } * @endcode */ connection_t make_subscriber() { auto conn = make_connection(); auto payload = make_payload("subscribe report"); if (conn->send(payload->data, payload->len, 0) == 0) throw system_error("Failed to initialize subscriber"); return conn; } } LEMONBUDDY_NS_END