diff --git a/README.md b/README.md
index 5266e916..4edf2253 100644
--- a/README.md
+++ b/README.md
@@ -116,6 +116,36 @@ sudo apt install \
libxkbregistry-dev
```
+On Arch, you can use this command:
+
+```
+pacman -S \
+ gtkmm3 \
+ jsoncpp \
+ libsigc++ \
+ fmt \
+ wayland \
+ chrono-date \
+ spdlog \
+ gtk3 \
+ gobject-introspection \
+ libgirepository \
+ libpulse \
+ libnl \
+ libappindicator-gtk3 \
+ libdbusmenu-gtk3 \
+ libmpdclient \
+ sndio \
+ libevdev \
+ libxkbcommon \
+ upower \
+ meson \
+ cmake \
+ scdoc \
+ wayland-protocols \
+ glib2-devel
+```
+
Contributions welcome!
Have fun :)
diff --git a/include/modules/network.hpp b/include/modules/network.hpp
index 5fd0c180..8e80d5aa 100644
--- a/include/modules/network.hpp
+++ b/include/modules/network.hpp
@@ -27,8 +27,8 @@ class Network : public ALabel {
auto update() -> void override;
private:
- static const uint8_t MAX_RETRY = 5;
- static const uint8_t EPOLL_MAX = 200;
+ static const uint8_t MAX_RETRY{5};
+ static const uint8_t EPOLL_MAX{200};
static int handleEvents(struct nl_msg*, void*);
static int handleEventsDone(struct nl_msg*, void*);
@@ -51,37 +51,37 @@ class Network : public ALabel {
bool wildcardMatch(const std::string& pattern, const std::string& text) const;
std::optional> readBandwidthUsage();
- int ifid_;
- ip_addr_pref addr_pref_;
- struct sockaddr_nl nladdr_ = {0};
- struct nl_sock* sock_ = nullptr;
- struct nl_sock* ev_sock_ = nullptr;
- int efd_;
- int ev_fd_;
- int nl80211_id_;
+ int ifid_{-1};
+ ip_addr_pref addr_pref_{ip_addr_pref::IPV4};
+ struct sockaddr_nl nladdr_{0};
+ struct nl_sock* sock_{nullptr};
+ struct nl_sock* ev_sock_{nullptr};
+ int efd_{-1};
+ int ev_fd_{-1};
+ int nl80211_id_{-1};
std::mutex mutex_;
- bool want_route_dump_;
- bool want_link_dump_;
- bool want_addr_dump_;
- bool dump_in_progress_;
- bool is_p2p_;
+ bool want_route_dump_{false};
+ bool want_link_dump_{false};
+ bool want_addr_dump_{false};
+ bool dump_in_progress_{false};
+ bool is_p2p_{false};
- unsigned long long bandwidth_down_total_;
- unsigned long long bandwidth_up_total_;
+ unsigned long long bandwidth_down_total_{0};
+ unsigned long long bandwidth_up_total_{0};
std::string state_;
std::string essid_;
std::string bssid_;
- bool carrier_;
+ bool carrier_{false};
std::string ifname_;
std::string ipaddr_;
std::string ipaddr6_;
std::string gwaddr_;
std::string netmask_;
std::string netmask6_;
- int cidr_;
- int cidr6_;
+ int cidr_{0};
+ int cidr6_{0};
int32_t signal_strength_dbm_;
uint8_t signal_strength_;
std::string signal_strength_app_;
@@ -90,9 +90,9 @@ class Network : public ALabel {
util::SleeperThread thread_;
util::SleeperThread thread_timer_;
#ifdef WANT_RFKILL
- util::Rfkill rfkill_;
+ util::Rfkill rfkill_{RFKILL_TYPE_WLAN};
#endif
- float frequency_;
+ float frequency_{0};
};
} // namespace waybar::modules
diff --git a/include/util/rfkill.hpp b/include/util/rfkill.hpp
index f620db53..47ab729b 100644
--- a/include/util/rfkill.hpp
+++ b/include/util/rfkill.hpp
@@ -5,6 +5,8 @@
#include
#include
+#include
+
namespace waybar::util {
class Rfkill : public sigc::trackable {
@@ -17,7 +19,7 @@ class Rfkill : public sigc::trackable {
private:
enum rfkill_type rfkill_type_;
- bool state_ = false;
+ std::atomic_bool state_ = false;
int fd_ = -1;
bool on_event(Glib::IOCondition cond);
diff --git a/man/waybar-network.5.scd b/man/waybar-network.5.scd
index 3b63e3ee..9e146ac0 100644
--- a/man/waybar-network.5.scd
+++ b/man/waybar-network.5.scd
@@ -16,6 +16,11 @@ Addressed by *network*
typeof: string ++
Use the defined interface instead of auto-detection. Accepts wildcard.
+*rfkill*: ++
+ typeof: bool ++
+ default: true ++
+ If enabled, the *disabled* format will be used when rfkill is blocking wlan interfaces.
+
*interval*: ++
typeof: integer ++
default: 60 ++
@@ -49,7 +54,7 @@ Addressed by *network*
*format-disabled*: ++
typeof: string ++
- This format is used when the displayed interface is disabled.
+ This format is used when rfkill is blocking wlan interfaces.
*format-icons*: ++
typeof: array/object ++
@@ -127,7 +132,7 @@ Addressed by *network*
*tooltip-format-disabled*: ++
typeof: string ++
- This format is used when the displayed interface is disabled.
+ This format is used when rfkill is blocking wlan interfaces.
*menu*: ++
typeof: string ++
@@ -157,7 +162,7 @@ Addressed by *network*
*{netmask}*: The subnetmask corresponding to the IP(V4).
-*{netmask6}*: The subnetmask corresponding to the IP(V6).
+*{netmask6}*: The subnetmask corresponding to the IP(V6).
*{cidr}*: The subnetmask corresponding to the IP(V4) in CIDR notation.
diff --git a/src/client.cpp b/src/client.cpp
index e363f236..946780db 100644
--- a/src/client.cpp
+++ b/src/client.cpp
@@ -151,15 +151,19 @@ void waybar::Client::handleDeferredMonitorRemoval(Glib::RefPtr mon
const std::string waybar::Client::getStyle(const std::string &style,
std::optional appearance = std::nullopt) {
+ auto gtk_settings = Gtk::Settings::get_default();
std::optional css_file;
+
if (style.empty()) {
std::vector search_files;
switch (appearance.value_or(portal->getAppearance())) {
case waybar::Appearance::LIGHT:
search_files.emplace_back("style-light.css");
+ gtk_settings->property_gtk_application_prefer_dark_theme() = false;
break;
case waybar::Appearance::DARK:
search_files.emplace_back("style-dark.css");
+ gtk_settings->property_gtk_application_prefer_dark_theme() = true;
break;
case waybar::Appearance::UNKNOWN:
break;
@@ -169,9 +173,11 @@ const std::string waybar::Client::getStyle(const std::string &style,
} else {
css_file = style;
}
+
if (!css_file) {
throw std::runtime_error("Missing required resource files");
}
+
spdlog::info("Using CSS file {}", css_file.value());
return css_file.value();
};
diff --git a/src/modules/battery.cpp b/src/modules/battery.cpp
index 32488d53..3d0da330 100644
--- a/src/modules/battery.cpp
+++ b/src/modules/battery.cpp
@@ -704,7 +704,7 @@ auto waybar::modules::Battery::update() -> void {
} else if (config_["tooltip-format"].isString()) {
tooltip_format = config_["tooltip-format"].asString();
}
- label_.set_tooltip_text(
+ label_.set_tooltip_markup(
fmt::format(fmt::runtime(tooltip_format), fmt::arg("timeTo", tooltip_text_default),
fmt::arg("power", power), fmt::arg("capacity", capacity),
fmt::arg("time", time_remaining_formatted), fmt::arg("cycles", cycles),
diff --git a/src/modules/hyprland/workspaces.cpp b/src/modules/hyprland/workspaces.cpp
index 3063231b..5d2903dc 100644
--- a/src/modules/hyprland/workspaces.cpp
+++ b/src/modules/hyprland/workspaces.cpp
@@ -66,14 +66,17 @@ Json::Value Workspaces::createMonitorWorkspaceData(std::string const &name,
void Workspaces::createWorkspace(Json::Value const &workspace_data,
Json::Value const &clients_data) {
auto workspaceName = workspace_data["name"].asString();
+ auto workspaceId = workspace_data["id"].asInt();
spdlog::debug("Creating workspace {}", workspaceName);
// avoid recreating existing workspaces
- auto workspace =
- std::ranges::find_if(m_workspaces, [workspaceName](std::unique_ptr const &w) {
- return (workspaceName.starts_with("special:") && workspaceName.substr(8) == w->name()) ||
- workspaceName == w->name();
- });
+ auto workspace = std::ranges::find_if(m_workspaces, [&](std::unique_ptr const &w) {
+ if (workspaceId > 0) {
+ return w->id() == workspaceId;
+ }
+ return (workspaceName.starts_with("special:") && workspaceName.substr(8) == w->name()) ||
+ workspaceName == w->name();
+ });
if (workspace != m_workspaces.end()) {
// don't recreate workspace, but update persistency if necessary
@@ -254,10 +257,8 @@ void Workspaces::loadPersistentWorkspacesFromConfig(Json::Value const &clientsJs
// value is an array => create defined workspaces for this monitor
if (canCreate) {
for (const Json::Value &workspace : value) {
- if (workspace.isInt()) {
- spdlog::debug("Creating workspace {} on monitor {}", workspace, currentMonitor);
- persistentWorkspacesToCreate.emplace_back(std::to_string(workspace.asInt()));
- }
+ spdlog::debug("Creating workspace {} on monitor {}", workspace, currentMonitor);
+ persistentWorkspacesToCreate.emplace_back(workspace.asString());
}
} else {
// key is the workspace and value is array of monitors to create on
@@ -293,8 +294,13 @@ void Workspaces::loadPersistentWorkspacesFromWorkspaceRules(const Json::Value &c
if (!rule["persistent"].asBool()) {
continue;
}
- auto const &workspace = rule.isMember("defaultName") ? rule["defaultName"].asString()
- : rule["workspaceString"].asString();
+ auto workspace = rule.isMember("defaultName") ? rule["defaultName"].asString()
+ : rule["workspaceString"].asString();
+
+ // The prefix "name:" cause mismatches with workspace names taken anywhere else.
+ if (workspace.starts_with("name:")) {
+ workspace = workspace.substr(5);
+ }
auto const &monitor = rule["monitor"].asString();
// create this workspace persistently if:
// 1. the allOutputs config option is enabled
@@ -1040,9 +1046,17 @@ bool Workspaces::updateWindowsToCreate() {
void Workspaces::updateWorkspaceStates() {
const std::vector visibleWorkspaces = getVisibleWorkspaces();
auto updatedWorkspaces = m_ipc.getSocket1JsonReply("workspaces");
+
+ auto currentWorkspace = m_ipc.getSocket1JsonReply("activeworkspace");
+ std::string currentWorkspaceName =
+ currentWorkspace.isMember("name") ? currentWorkspace["name"].asString() : "";
+
for (auto &workspace : m_workspaces) {
+ bool isActiveByName =
+ !currentWorkspaceName.empty() && workspace->name() == currentWorkspaceName;
+
workspace->setActive(
- workspace->id() == m_activeWorkspaceId ||
+ workspace->id() == m_activeWorkspaceId || isActiveByName ||
(workspace->isSpecial() && workspace->name() == m_activeSpecialWorkspaceName));
if (workspace->isActive() && workspace->isUrgent()) {
workspace->setUrgent(false);
@@ -1120,4 +1134,4 @@ std::optional Workspaces::parseWorkspaceId(std::string const &workspaceIdSt
}
}
-} // namespace waybar::modules::hyprland
+} // namespace waybar::modules::hyprland
\ No newline at end of file
diff --git a/src/modules/network.cpp b/src/modules/network.cpp
index cc096f72..94c77a69 100644
--- a/src/modules/network.cpp
+++ b/src/modules/network.cpp
@@ -78,25 +78,7 @@ waybar::modules::Network::readBandwidthUsage() {
}
waybar::modules::Network::Network(const std::string &id, const Json::Value &config)
- : ALabel(config, "network", id, DEFAULT_FORMAT, 60),
- ifid_(-1),
- addr_pref_(IPV4),
- efd_(-1),
- ev_fd_(-1),
- want_route_dump_(false),
- want_link_dump_(false),
- want_addr_dump_(false),
- dump_in_progress_(false),
- is_p2p_(false),
- cidr_(0),
- cidr6_(0),
- signal_strength_dbm_(0),
- signal_strength_(0),
-#ifdef WANT_RFKILL
- rfkill_{RFKILL_TYPE_WLAN},
-#endif
- frequency_(0.0) {
-
+ : ALabel(config, "network", id, DEFAULT_FORMAT, 60) {
// Start with some "text" in the module's label_. update() will then
// update it. Since the text should be different, update() will be able
// to show or hide the event_box_. This is to work around the case where
@@ -271,11 +253,16 @@ void waybar::modules::Network::worker() {
}
const std::string waybar::modules::Network::getNetworkState() const {
+ if (ifid_ == -1 || !carrier_) {
#ifdef WANT_RFKILL
- if (rfkill_.getState() && ifid_ == -1) return "disabled";
+ bool display_rfkill = true;
+ if (config_["rfkill"].isBool()) {
+ display_rfkill = config_["rfkill"].asBool();
+ }
+ if (rfkill_.getState() && display_rfkill) return "disabled";
#endif
- if (ifid_ == -1) return "disconnected";
- if (!carrier_) return "disconnected";
+ return "disconnected";
+ }
if (ipaddr_.empty() && ipaddr6_.empty()) return "linked";
if (essid_.empty()) return "ethernet";
return "wifi";
@@ -418,6 +405,7 @@ void waybar::modules::Network::clearIface() {
netmask_.clear();
netmask6_.clear();
carrier_ = false;
+ is_p2p_ = false;
cidr_ = 0;
cidr6_ = 0;
signal_strength_dbm_ = 0;
@@ -507,6 +495,11 @@ int waybar::modules::Network::handleEvents(struct nl_msg *msg, void *data) {
net->ifname_ = new_ifname;
net->ifid_ = ifi->ifi_index;
if (ifi->ifi_flags & IFF_POINTOPOINT) net->is_p2p_ = true;
+ if ((ifi->ifi_flags & IFF_UP) == 0) {
+ // With some network drivers (e.g. mt7921e), the interface may
+ // report having a carrier even though interface is down.
+ carrier = false;
+ }
if (carrier.has_value()) {
net->carrier_ = carrier.value();
}
diff --git a/src/modules/pulseaudio.cpp b/src/modules/pulseaudio.cpp
index 255ca571..ceed20dd 100644
--- a/src/modules/pulseaudio.cpp
+++ b/src/modules/pulseaudio.cpp
@@ -132,7 +132,7 @@ auto waybar::modules::Pulseaudio::update() -> void {
tooltip_format = config_["tooltip-format"].asString();
}
if (!tooltip_format.empty()) {
- label_.set_tooltip_text(fmt::format(
+ label_.set_tooltip_markup(fmt::format(
fmt::runtime(tooltip_format), fmt::arg("desc", sink_desc),
fmt::arg("volume", sink_volume), fmt::arg("format_source", format_source),
fmt::arg("source_volume", source_volume), fmt::arg("source_desc", source_desc),
diff --git a/src/modules/sni/item.cpp b/src/modules/sni/item.cpp
index 407d7e72..4e80eba7 100644
--- a/src/modules/sni/item.cpp
+++ b/src/modules/sni/item.cpp
@@ -140,7 +140,25 @@ void Item::setProperty(const Glib::ustring& name, Glib::VariantBase& value) {
category = get_variant(value);
} else if (name == "Id") {
id = get_variant(value);
- setCustomIcon(id);
+
+ /*
+ * HACK: Electron apps seem to have the same ID, but tooltip seems correct, so use that as ID
+ * to pass as the custom icon option. I'm avoiding being disruptive and setting that to the ID
+ * itself as I've no idea what this would affect.
+ * The tooltip text is converted to lowercase since that's what (most?) themes expect?
+ * I still haven't found a way for it to pick from theme automatically, although
+ * it might be my theme.
+ */
+ if (id == "chrome_status_icon_1") {
+ Glib::VariantBase value;
+ this->proxy_->get_cached_property(value, "ToolTip");
+ tooltip = get_variant(value);
+ if (!tooltip.text.empty()) {
+ setCustomIcon(tooltip.text.lowercase());
+ }
+ } else {
+ setCustomIcon(id);
+ }
} else if (name == "Title") {
title = get_variant(value);
if (tooltip.text.empty()) {
@@ -203,6 +221,8 @@ void Item::setStatus(const Glib::ustring& value) {
}
void Item::setCustomIcon(const std::string& id) {
+ spdlog::debug("SNI tray id: {}", id);
+
std::string custom_icon = IconManager::instance().getIconForApp(id);
if (!custom_icon.empty()) {
if (std::filesystem::exists(custom_icon)) {
diff --git a/src/modules/upower.cpp b/src/modules/upower.cpp
index 4b832b7e..4844582f 100644
--- a/src/modules/upower.cpp
+++ b/src/modules/upower.cpp
@@ -10,7 +10,6 @@ UPower::UPower(const std::string &id, const Json::Value &config)
: AIconLabel(config, "upower", id, "{percentage}", 0, true, true, true), sleeping_{false} {
box_.set_name(name_);
box_.set_spacing(0);
- box_.set_has_tooltip(AModule::tooltipEnabled());
// Tooltip box
contentBox_.set_orientation((box_.get_orientation() == Gtk::ORIENTATION_HORIZONTAL)
? Gtk::ORIENTATION_VERTICAL
@@ -70,8 +69,10 @@ UPower::UPower(const std::string &id, const Json::Value &config)
g_signal_connect(upClient_, "device-removed", G_CALLBACK(deviceRemoved_cb), this);
// Subscribe tooltip query events
- box_.set_has_tooltip();
- box_.signal_query_tooltip().connect(sigc::mem_fun(*this, &UPower::queryTooltipCb), false);
+ box_.set_has_tooltip(AModule::tooltipEnabled());
+ if (AModule::tooltipEnabled()) {
+ box_.signal_query_tooltip().connect(sigc::mem_fun(*this, &UPower::queryTooltipCb), false);
+ }
resetDevices();
setDisplayDevice();