mirror of
				https://github.com/rad4day/Waybar.git
				synced 2025-10-26 14:32:28 +01:00 
			
		
		
		
	Add DWL tags module
This commit is contained in:
		
							
								
								
									
										224
									
								
								src/modules/dwl/tags.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										224
									
								
								src/modules/dwl/tags.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,224 @@ | ||||
| #include "modules/dwl/tags.hpp" | ||||
|  | ||||
| #include <gtkmm/button.h> | ||||
| #include <gtkmm/label.h> | ||||
| #include <spdlog/spdlog.h> | ||||
| #include <wayland-client.h> | ||||
|  | ||||
| #include <algorithm> | ||||
|  | ||||
| #include "client.hpp" | ||||
| #include "dwl-bar-ipc-unstable-v1-client-protocol.h" | ||||
|  | ||||
| #define TAG_INACTIVE 0 | ||||
| #define TAG_ACTIVE 1 | ||||
| #define TAG_URGENT 2 | ||||
|  | ||||
| namespace waybar::modules::dwl { | ||||
|  | ||||
| /* dwl stuff */ | ||||
| wl_array tags, layouts; | ||||
|  | ||||
| static uint num_tags = 0; | ||||
|  | ||||
| void toggle_visibility(void* data, zdwl_output_v1* zdwl_output_v1) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| void active(void* data, zdwl_output_v1* zdwl_output_v1, uint32_t active) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| static void set_tag(void* data, zdwl_output_v1* zdwl_output_v1, uint32_t tag, uint32_t state, uint32_t clients, uint32_t focused) { | ||||
|   static_cast<Tags *>(data)->handle_view_tags(tag, state, clients, focused); | ||||
|  | ||||
|   num_tags = (state & ZDWL_OUTPUT_V1_TAG_STATE_ACTIVE) ? num_tags | (1 << tag) : num_tags & ~(1 << tag); | ||||
| } | ||||
|  | ||||
| void set_layout_symbol(void* data, zdwl_output_v1* zdwl_output_v1, const char *layout) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| void title(void* data, zdwl_output_v1* zdwl_output_v1, const char* title) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| void dwl_frame(void* data, zdwl_output_v1* zdwl_output_v1) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| static void set_layout(void* data, zdwl_output_v1* zdwl_output_v1, uint32_t layout) { | ||||
|   // Intentionally empty | ||||
| } | ||||
|  | ||||
| static void appid(void *data, zdwl_output_v1 *zdwl_output_v1, const char *appid) { | ||||
|   // Intentionally empty | ||||
| }; | ||||
|  | ||||
| static const zdwl_output_v1_listener output_status_listener_impl { | ||||
|     .toggle_visibility = toggle_visibility, | ||||
|     .active = active, | ||||
|     .tag = set_tag, | ||||
|     .layout = set_layout, | ||||
|     .title = title, | ||||
|     .appid = appid, | ||||
|     .layout_symbol = set_layout_symbol, | ||||
|     .frame = dwl_frame, | ||||
| }; | ||||
|  | ||||
| void add_layout(void* data, zdwl_manager_v1* zdwl_manager_v1, const char* name) { | ||||
|   void* temp = wl_array_add(&layouts, sizeof(char**)); | ||||
|   if (!temp) | ||||
|     return; | ||||
|  | ||||
|   char* dup = strdup(name); | ||||
|  | ||||
|   memcpy(temp, &dup, sizeof(char**)); | ||||
| } | ||||
|  | ||||
| void add_tag(void* data, zdwl_manager_v1* zdwl_manager_v1, const char* name) { | ||||
|   void* temp = wl_array_add(&tags, sizeof(char**)); | ||||
|   if (!temp) | ||||
|     return; | ||||
|  | ||||
|   char* dup = strdup(name); /* Gain ownership of name */ | ||||
|  | ||||
|   memcpy(temp, &dup, sizeof(char**)); /* Copy a pointer of it into the array */; | ||||
| } | ||||
|  | ||||
| static const struct zdwl_manager_v1_listener dwl_listener = { | ||||
|   .tag = add_tag, | ||||
|   .layout = add_layout, | ||||
| }; | ||||
|  | ||||
| static void handle_global(void *data, struct wl_registry *registry, uint32_t name, | ||||
|                           const char *interface, uint32_t version) { | ||||
|   if (std::strcmp(interface, zdwl_manager_v1_interface.name) == 0) { | ||||
|     static_cast<Tags *>(data)->status_manager_ = static_cast<struct zdwl_manager_v1 *>( | ||||
|         (zdwl_manager_v1*)wl_registry_bind(registry, name, &zdwl_manager_v1_interface, 3)); | ||||
|         zdwl_manager_v1_add_listener(static_cast<Tags *>(data)->status_manager_, &dwl_listener, NULL); | ||||
|   } | ||||
|   if (std::strcmp(interface, wl_seat_interface.name) == 0) { | ||||
|     version = std::min<uint32_t>(version, 1); | ||||
|     static_cast<Tags *>(data)->seat_ = static_cast<struct wl_seat *>( | ||||
|         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}, | ||||
|       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("dwl_status_manager_v1 not advertised"); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   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<uint32_t>(32, config_["num-tags"].asUInt()) : 9; | ||||
|  | ||||
|   std::vector<std::string> 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_ = zdwl_manager_v1_get_output(status_manager_, output); | ||||
|   zdwl_output_v1_add_listener(output_status_, &output_status_listener_impl, this); | ||||
|  | ||||
|   zdwl_manager_v1_destroy(status_manager_); | ||||
| } | ||||
|  | ||||
| Tags::~Tags() { | ||||
|   if (output_status_) { | ||||
|     zdwl_manager_v1_destroy(status_manager_); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void Tags::handle_primary_clicked(uint32_t tag) { | ||||
|   if (!output_status_) return; | ||||
|  | ||||
|   zdwl_output_v1_set_tags(output_status_, tag, 1); | ||||
| } | ||||
|  | ||||
| bool Tags::handle_button_press(GdkEventButton *event_button, uint32_t tag) { | ||||
|   if (event_button->type == GDK_BUTTON_PRESS && event_button->button == 3) { | ||||
|  | ||||
|     if (!output_status_) return true; | ||||
|     zdwl_output_v1_set_tags(output_status_, num_tags ^ tag, 0); | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| void Tags::handle_view_tags(uint32_t tag, uint32_t state, uint32_t clients, uint32_t focused) { | ||||
|   // First clear all occupied state | ||||
|   auto &button = buttons_[tag]; | ||||
|   if (clients) { | ||||
|     button.get_style_context()->add_class("occupied"); | ||||
|   } else { | ||||
|     button.get_style_context()->remove_class("occupied"); | ||||
|   } | ||||
|  | ||||
|   if (state & TAG_ACTIVE) { | ||||
|       button.get_style_context()->add_class("focused"); | ||||
|   } else { | ||||
|       button.get_style_context()->remove_class("focused"); | ||||
|   } | ||||
|  | ||||
|   if (state & TAG_URGENT) { | ||||
|       button.get_style_context()->add_class("urgent"); | ||||
|   } else { | ||||
|       button.get_style_context()->remove_class("urgent"); | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
| } /* namespace waybar::modules::dwl */ | ||||
		Reference in New Issue
	
	Block a user
	 David Delarosa
					David Delarosa