#include "modules/river/tags.hpp" #include #include #include #include #include #include "client.hpp" #include "xdg-output-unstable-v1-client-protocol.h" namespace waybar::modules::river { static void listen_focused_tags(void *data, struct zriver_output_status_v1 *zriver_output_status_v1, uint32_t tags) { static_cast(data)->handle_focused_tags(tags); } static void listen_view_tags(void *data, struct zriver_output_status_v1 *zriver_output_status_v1, struct wl_array *tags) { static_cast(data)->handle_view_tags(tags); } static void listen_urgent_tags(void *data, struct zriver_output_status_v1 *zriver_output_status_v1, uint32_t tags) { static_cast(data)->handle_urgent_tags(tags); } static const zriver_output_status_v1_listener output_status_listener_impl{ .focused_tags = listen_focused_tags, .view_tags = listen_view_tags, .urgent_tags = listen_urgent_tags, }; static void listen_command_success(void *data, struct zriver_command_callback_v1 *zriver_command_callback_v1, const char *output) { // Do nothing but keep listener to avoid crashing when command was successful } static void listen_command_failure(void *data, struct zriver_command_callback_v1 *zriver_command_callback_v1, const char *output) { spdlog::error("failure when selecting/toggling tags {}", output); } static const zriver_command_callback_v1_listener command_callback_listener_impl{ .success = listen_command_success, .failure = listen_command_failure, }; static void handle_global(void *data, struct wl_registry *registry, uint32_t name, const char *interface, uint32_t version) { if (std::strcmp(interface, zriver_status_manager_v1_interface.name) == 0) { version = std::min(version, 2); if (version < ZRIVER_OUTPUT_STATUS_V1_URGENT_TAGS_SINCE_VERSION) { spdlog::warn("river server does not support urgent tags"); } static_cast(data)->status_manager_ = static_cast( wl_registry_bind(registry, name, &zriver_status_manager_v1_interface, version)); } if (std::strcmp(interface, zriver_control_v1_interface.name) == 0) { version = std::min(version, 1); static_cast(data)->control_ = static_cast( wl_registry_bind(registry, name, &zriver_control_v1_interface, version)); } if (std::strcmp(interface, wl_seat_interface.name) == 0) { version = std::min(version, 1); static_cast(data)->seat_ = static_cast( wl_registry_bind(registry, name, &wl_seat_interface, version)); } } static void handle_global_remove(void *data, struct wl_registry *registry, uint32_t name) { /* Ignore event */ } static const wl_registry_listener registry_listener_impl = {.global = handle_global, .global_remove = handle_global_remove}; Tags::Tags(const std::string &id, const waybar::Bar &bar, const Json::Value &config) : waybar::AModule(config, "tags", id, false, false), status_manager_{nullptr}, control_{nullptr}, seat_{nullptr}, bar_(bar), box_{bar.vertical ? Gtk::ORIENTATION_VERTICAL : Gtk::ORIENTATION_HORIZONTAL, 0}, output_status_{nullptr} { struct wl_display *display = Client::inst()->wl_display; struct wl_registry *registry = wl_display_get_registry(display); wl_registry_add_listener(registry, ®istry_listener_impl, this); wl_display_roundtrip(display); if (!status_manager_) { spdlog::error("river_status_manager_v1 not advertised"); return; } if (!control_) { spdlog::error("river_control_v1 not advertised"); } if (!seat_) { spdlog::error("wl_seat not advertised"); } box_.set_name("tags"); if (!id.empty()) { box_.get_style_context()->add_class(id); } event_box_.add(box_); // Default to 9 tags, cap at 32 const uint32_t num_tags = config["num-tags"].isUInt() ? std::min(32, config_["num-tags"].asUInt()) : 9; std::vector tag_labels(num_tags); for (uint32_t tag = 0; tag < num_tags; ++tag) { tag_labels[tag] = std::to_string(tag + 1); } const Json::Value custom_labels = config["tag-labels"]; if (custom_labels.isArray() && !custom_labels.empty()) { for (uint32_t tag = 0; tag < std::min(num_tags, custom_labels.size()); ++tag) { tag_labels[tag] = custom_labels[tag].asString(); } } uint32_t i = 1; for (const auto &tag_label : tag_labels) { Gtk::Button &button = buttons_.emplace_back(tag_label); button.set_relief(Gtk::RELIEF_NONE); box_.pack_start(button, false, false, 0); if (!config_["disable-click"].asBool()) { button.signal_clicked().connect( sigc::bind(sigc::mem_fun(*this, &Tags::handle_primary_clicked), i)); button.signal_button_press_event().connect( sigc::bind(sigc::mem_fun(*this, &Tags::handle_button_press), i)); } button.show(); i <<= 1; } struct wl_output *output = gdk_wayland_monitor_get_wl_output(bar_.output->monitor->gobj()); output_status_ = zriver_status_manager_v1_get_river_output_status(status_manager_, output); zriver_output_status_v1_add_listener(output_status_, &output_status_listener_impl, this); zriver_status_manager_v1_destroy(status_manager_); } Tags::~Tags() { if (output_status_) { zriver_output_status_v1_destroy(output_status_); } if (control_) { zriver_control_v1_destroy(control_); } } void Tags::handle_primary_clicked(uint32_t tag) { // Send river command to select tag on left mouse click zriver_command_callback_v1 *callback; zriver_control_v1_add_argument(control_, "set-focused-tags"); zriver_control_v1_add_argument(control_, std::to_string(tag).c_str()); callback = zriver_control_v1_run_command(control_, seat_); zriver_command_callback_v1_add_listener(callback, &command_callback_listener_impl, nullptr); } bool Tags::handle_button_press(GdkEventButton *event_button, uint32_t tag) { if (event_button->type == GDK_BUTTON_PRESS && event_button->button == 3) { // Send river command to toggle tag on right mouse click zriver_command_callback_v1 *callback; zriver_control_v1_add_argument(control_, "toggle-focused-tags"); zriver_control_v1_add_argument(control_, std::to_string(tag).c_str()); callback = zriver_control_v1_run_command(control_, seat_); zriver_command_callback_v1_add_listener(callback, &command_callback_listener_impl, nullptr); } return true; } void Tags::handle_focused_tags(uint32_t tags) { uint32_t i = 0; for (auto &button : buttons_) { if ((1 << i) & tags) { button.get_style_context()->add_class("focused"); } else { button.get_style_context()->remove_class("focused"); } ++i; } } void Tags::handle_view_tags(struct wl_array *view_tags) { // First clear all occupied state for (auto &button : buttons_) { button.get_style_context()->remove_class("occupied"); } // Set tags with a view to occupied uint32_t *start = static_cast(view_tags->data); for (uint32_t *tags = start; tags < start + view_tags->size / sizeof(uint32_t); ++tags) { uint32_t i = 0; for (auto &button : buttons_) { if (*tags & (1 << i)) { button.get_style_context()->add_class("occupied"); } ++i; } } } void Tags::handle_urgent_tags(uint32_t tags) { uint32_t i = 0; for (auto &button : buttons_) { if ((1 << i) & tags) { button.get_style_context()->add_class("urgent"); } else { button.get_style_context()->remove_class("urgent"); } ++i; } } } /* namespace waybar::modules::river */