mirror of
				https://github.com/rad4day/Waybar.git
				synced 2025-10-25 23:22:28 +02:00 
			
		
		
		
	Merge branch 'master' into on-update
This commit is contained in:
		| @@ -115,6 +115,16 @@ const std::tuple<uint8_t, float, std::string> waybar::modules::Battery::getInfos | ||||
|       time_remaining = -(float)(total_energy_full - total_energy) / total_power; | ||||
|     } | ||||
|     uint16_t capacity = total / batteries_.size(); | ||||
|     // Handle full-at | ||||
|     if (config_["full-at"].isUInt()) { | ||||
|       auto full_at = config_["full-at"].asUInt(); | ||||
|       if (full_at < 100) { | ||||
|         capacity = static_cast<float>(capacity / full_at) * 100; | ||||
|         if (capacity > full_at) { | ||||
|           capacity = full_at; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     return {capacity, time_remaining, status}; | ||||
|   } catch (const std::exception& e) { | ||||
|     spdlog::error("Battery: {}", e.what()); | ||||
| @@ -163,7 +173,12 @@ auto waybar::modules::Battery::update() -> void { | ||||
|     } | ||||
|     label_.set_tooltip_text(tooltip_text); | ||||
|   } | ||||
|   // Transform to lowercase | ||||
|   std::transform(status.begin(), status.end(), status.begin(), ::tolower); | ||||
|   // Replace space with dash | ||||
|   std::transform(status.begin(), status.end(), status.begin(), [](char ch) { | ||||
|     return ch == ' ' ? '-' : ch; | ||||
|   }); | ||||
|   auto format = format_; | ||||
|   auto state = getState(capacity, true); | ||||
|   if (!old_status_.empty()) { | ||||
|   | ||||
| @@ -1,8 +1,28 @@ | ||||
| #include "modules/clock.hpp" | ||||
| #include <time.h> | ||||
| #include <sstream> | ||||
| #include <type_traits> | ||||
| #ifdef HAVE_LANGINFO_1STDAY | ||||
| #include <langinfo.h> | ||||
| #include <locale.h> | ||||
| #endif | ||||
|  | ||||
| using waybar::modules::waybar_time; | ||||
|  | ||||
| waybar::modules::Clock::Clock(const std::string& id, const Json::Value& config) | ||||
|     : ALabel(config, "clock", id, "{:%H:%M}", 60) { | ||||
|     : ALabel(config, "clock", id, "{:%H:%M}", 60) | ||||
|     , fixed_time_zone_(false) | ||||
| { | ||||
|   if (config_["timezone"].isString()) { | ||||
|     time_zone_ = date::locate_zone(config_["timezone"].asString()); | ||||
|     fixed_time_zone_ = true; | ||||
|   } | ||||
|  | ||||
|   if (config_["locale"].isString()) { | ||||
|     locale_ = std::locale(config_["locale"].asString()); | ||||
|   } else { | ||||
|     locale_ = std::locale(""); | ||||
|   } | ||||
|  | ||||
|   thread_ = [this] { | ||||
|     dp.emit(); | ||||
|     auto now = std::chrono::system_clock::now(); | ||||
| @@ -13,21 +33,115 @@ waybar::modules::Clock::Clock(const std::string& id, const Json::Value& config) | ||||
| } | ||||
|  | ||||
| auto waybar::modules::Clock::update() -> void { | ||||
|   tzset(); // Update timezone information | ||||
|   auto now = std::chrono::system_clock::now(); | ||||
|   auto localtime = fmt::localtime(std::chrono::system_clock::to_time_t(now)); | ||||
|   auto text = fmt::format(format_, localtime); | ||||
|   if (!fixed_time_zone_) { | ||||
|     // Time zone can change. Be sure to pick that. | ||||
|     time_zone_ = date::current_zone(); | ||||
|   } | ||||
|   auto        now = std::chrono::system_clock::now(); | ||||
|   waybar_time wtime = {locale_, | ||||
|                        date::make_zoned(time_zone_, date::floor<std::chrono::seconds>(now))}; | ||||
|  | ||||
|   auto text = fmt::format(format_, wtime); | ||||
|   label_.set_markup(text); | ||||
|  | ||||
|   if (tooltipEnabled()) { | ||||
|     if (config_["tooltip-format"].isString()) { | ||||
|       const auto calendar = calendar_text(wtime); | ||||
|       auto tooltip_format = config_["tooltip-format"].asString(); | ||||
|       auto tooltip_text = fmt::format(tooltip_format, localtime); | ||||
|       label_.set_tooltip_text(tooltip_text); | ||||
|       auto tooltip_text = fmt::format(tooltip_format, wtime, fmt::arg("calendar", calendar)); | ||||
|       label_.set_tooltip_markup(tooltip_text); | ||||
|     } else { | ||||
|       label_.set_tooltip_text(text); | ||||
|       label_.set_tooltip_markup(text); | ||||
|     } | ||||
|   } | ||||
|   // Call parent update | ||||
|   ALabel::update(); | ||||
| } | ||||
|  | ||||
| auto waybar::modules::Clock::calendar_text(const waybar_time& wtime) -> std::string { | ||||
|   const auto daypoint = date::floor<date::days>(wtime.ztime.get_local_time()); | ||||
|   const auto ymd = date::year_month_day(daypoint); | ||||
|   if (cached_calendar_ymd_ == ymd) { | ||||
|     return cached_calendar_text_; | ||||
|   } | ||||
|  | ||||
|   const date::year_month ym(ymd.year(), ymd.month()); | ||||
|   const auto curr_day = ymd.day(); | ||||
|  | ||||
|   std::stringstream os; | ||||
|   const auto first_dow = first_day_of_week(); | ||||
|   weekdays_header(first_dow, os); | ||||
|  | ||||
|   // First week prefixed with spaces if needed. | ||||
|   auto wd = date::weekday(ym/1); | ||||
|   auto empty_days = (wd - first_dow).count(); | ||||
|   if (empty_days > 0) { | ||||
|     os << std::string(empty_days * 3 - 1, ' '); | ||||
|   } | ||||
|   auto last_day = (ym/date::literals::last).day(); | ||||
|   for (auto d = date::day(1); d <= last_day; ++d, ++wd) { | ||||
|     if (wd != first_dow) { | ||||
|       os << ' '; | ||||
|     } else if (unsigned(d) != 1) { | ||||
|       os << '\n'; | ||||
|     } | ||||
|     if (d == curr_day) { | ||||
|       os << "<b><u>" << date::format("%e", d) << "</u></b>"; | ||||
|     } else { | ||||
|       os << date::format("%e", d); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   auto result = os.str(); | ||||
|   cached_calendar_ymd_ = ymd; | ||||
|   cached_calendar_text_ = result; | ||||
|   return result; | ||||
| } | ||||
|  | ||||
| auto waybar::modules::Clock::weekdays_header(const date::weekday& first_dow, std::ostream& os) -> void { | ||||
|   auto wd = first_dow; | ||||
|   do { | ||||
|     if (wd != first_dow) os << ' '; | ||||
|     Glib::ustring wd_ustring(date::format(locale_, "%a", wd)); | ||||
|     auto wd_len = wd_ustring.length(); | ||||
|     if (wd_len > 2) { | ||||
|       wd_ustring = wd_ustring.substr(0, 2); | ||||
|       wd_len = 2; | ||||
|     } | ||||
|     const std::string pad(2 - wd_len, ' '); | ||||
|     os << pad << wd_ustring; | ||||
|   } while (++wd != first_dow); | ||||
|   os << "\n"; | ||||
| } | ||||
|  | ||||
| #ifdef HAVE_LANGINFO_1STDAY | ||||
| template <auto fn> | ||||
| using deleter_from_fn = std::integral_constant<decltype(fn), fn>; | ||||
|  | ||||
| template <typename T, auto fn> | ||||
| using deleting_unique_ptr = std::unique_ptr<T, deleter_from_fn<fn>>; | ||||
| #endif | ||||
|  | ||||
| // Computations done similarly to Linux cal utility. | ||||
| auto waybar::modules::Clock::first_day_of_week() -> date::weekday { | ||||
| #ifdef HAVE_LANGINFO_1STDAY | ||||
|   deleting_unique_ptr<std::remove_pointer<locale_t>::type, freelocale> | ||||
|     posix_locale{newlocale(LC_ALL, locale_.name().c_str(), nullptr)}; | ||||
|   if (posix_locale) { | ||||
|     const int i = (std::intptr_t) nl_langinfo_l(_NL_TIME_WEEK_1STDAY, posix_locale.get()); | ||||
|     auto ymd = date::year(i / 10000)/(i / 100 % 100)/(i % 100); | ||||
|     auto wd = date::weekday(ymd); | ||||
|     uint8_t j = *nl_langinfo_l(_NL_TIME_FIRST_WEEKDAY, posix_locale.get()); | ||||
|     return wd + date::days(j - 1); | ||||
|   } | ||||
| #endif | ||||
|   return date::Sunday; | ||||
| } | ||||
|  | ||||
| template <> | ||||
| struct fmt::formatter<waybar_time> : fmt::formatter<std::tm> { | ||||
|   template <typename FormatContext> | ||||
|   auto format(const waybar_time& t, FormatContext& ctx) { | ||||
|     return format_to(ctx.out(), "{}", date::format(t.locale, fmt::to_string(tm_format), t.ztime)); | ||||
|   } | ||||
| }; | ||||
|   | ||||
| @@ -49,19 +49,24 @@ void waybar::modules::Custom::continuousWorker() { | ||||
|   thread_ = [&] { | ||||
|     char*  buff = nullptr; | ||||
|     size_t len = 0; | ||||
|     bool restart = false; | ||||
|     if (getline(&buff, &len, fp_) == -1) { | ||||
|       int exit_code = 1; | ||||
|       if (fp_) { | ||||
|         exit_code = WEXITSTATUS(util::command::close(fp_, pid_)); | ||||
|         fp_ = nullptr; | ||||
|       } | ||||
|       thread_.stop(); | ||||
|       if (exit_code != 0) { | ||||
|         output_ = {exit_code, ""}; | ||||
|         dp.emit(); | ||||
|         spdlog::error("{} stopped unexpectedly, is it endless?", name_); | ||||
|       } | ||||
|       return; | ||||
|       if (config_["restart-interval"].isUInt()) { | ||||
|         restart = true; | ||||
|       } else { | ||||
|         thread_.stop(); | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|     std::string output = buff; | ||||
|  | ||||
| @@ -71,6 +76,14 @@ void waybar::modules::Custom::continuousWorker() { | ||||
|     } | ||||
|     output_ = {0, output}; | ||||
|     dp.emit(); | ||||
|     if (restart) { | ||||
|       pid_ = -1; | ||||
|       fp_ = util::command::open(cmd, pid_); | ||||
|       if (!fp_) { | ||||
|         throw std::runtime_error("Unable to open " + cmd); | ||||
|       } | ||||
|       thread_.sleep_for(std::chrono::seconds(config_["restart-interval"].asUInt())); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -10,11 +10,23 @@ waybar::modules::Memory::Memory(const std::string& id, const Json::Value& config | ||||
|  | ||||
| auto waybar::modules::Memory::update() -> void { | ||||
|   parseMeminfo(); | ||||
|   if (memtotal_ > 0 && memfree_ >= 0) { | ||||
|     auto total_ram_gigabytes = memtotal_ / std::pow(1024, 2); | ||||
|     int  used_ram_percentage = 100 * (memtotal_ - memfree_) / memtotal_; | ||||
|     auto used_ram_gigabytes = (memtotal_ - memfree_) / std::pow(1024, 2); | ||||
|     auto available_ram_gigabytes = memfree_ / std::pow(1024, 2); | ||||
|  | ||||
|   unsigned long memtotal = meminfo_["MemTotal"]; | ||||
|   unsigned long memfree; | ||||
|   if (meminfo_.count("MemAvailable")) { | ||||
|     // New kernels (3.4+) have an accurate available memory field. | ||||
|     memfree = meminfo_["MemAvailable"]; | ||||
|   } else { | ||||
|     // Old kernel; give a best-effort approximation of available memory. | ||||
|     memfree = meminfo_["MemFree"] + meminfo_["Buffers"] + meminfo_["Cached"] + | ||||
|               meminfo_["SReclaimable"] - meminfo_["Shmem"]; | ||||
|   } | ||||
|  | ||||
|   if (memtotal > 0 && memfree >= 0) { | ||||
|     auto total_ram_gigabytes = memtotal / std::pow(1024, 2); | ||||
|     int  used_ram_percentage = 100 * (memtotal - memfree) / memtotal; | ||||
|     auto used_ram_gigabytes = (memtotal - memfree) / std::pow(1024, 2); | ||||
|     auto available_ram_gigabytes = memfree / std::pow(1024, 2); | ||||
|  | ||||
|     getState(used_ram_percentage); | ||||
|     label_.set_markup(fmt::format(format_, | ||||
| @@ -35,7 +47,6 @@ auto waybar::modules::Memory::update() -> void { | ||||
| } | ||||
|  | ||||
| void waybar::modules::Memory::parseMeminfo() { | ||||
|   int64_t       memfree = -1, membuffer = -1, memcache = -1, memavail = -1; | ||||
|   std::ifstream info(data_dir_); | ||||
|   if (!info.is_open()) { | ||||
|     throw std::runtime_error("Can't open " + data_dir_); | ||||
| @@ -46,23 +57,9 @@ void waybar::modules::Memory::parseMeminfo() { | ||||
|     if (posDelim == std::string::npos) { | ||||
|       continue; | ||||
|     } | ||||
|  | ||||
|     std::string name = line.substr(0, posDelim); | ||||
|     int64_t     value = std::stol(line.substr(posDelim + 1)); | ||||
|  | ||||
|     if (name.compare("MemTotal") == 0) { | ||||
|       memtotal_ = value; | ||||
|     } else if (name.compare("MemAvailable") == 0) { | ||||
|       memavail = value; | ||||
|     } else if (name.compare("MemFree") == 0) { | ||||
|       memfree = value; | ||||
|     } else if (name.compare("Buffers") == 0) { | ||||
|       membuffer = value; | ||||
|     } else if (name.compare("Cached") == 0) { | ||||
|       memcache = value; | ||||
|     } | ||||
|     if (memtotal_ > 0 && (memavail >= 0 || (memfree > -1 && membuffer > -1 && memcache > -1))) { | ||||
|       break; | ||||
|     } | ||||
|     meminfo_[name] = value; | ||||
|   } | ||||
|   memfree_ = memavail >= 0 ? memavail : memfree + membuffer + memcache; | ||||
| } | ||||
|   | ||||
| @@ -143,7 +143,9 @@ void waybar::modules::MPD::setLabel() { | ||||
|     if (playing()) { | ||||
|       label_.get_style_context()->add_class("playing"); | ||||
|       label_.get_style_context()->remove_class("paused"); | ||||
|     } else { | ||||
|     } else if (paused()) { | ||||
|       format = | ||||
|         config_["format-paused"].isString() ? config_["format-paused"].asString() : config_["format"].asString(); | ||||
|       label_.get_style_context()->add_class("paused"); | ||||
|       label_.get_style_context()->remove_class("playing"); | ||||
|     } | ||||
| @@ -349,3 +351,5 @@ bool waybar::modules::MPD::stopped() { | ||||
| } | ||||
|  | ||||
| bool waybar::modules::MPD::playing() { return connection_ != nullptr && state_ == MPD_STATE_PLAY; } | ||||
|  | ||||
| bool waybar::modules::MPD::paused() { return connection_ != nullptr && state_ == MPD_STATE_PAUSE; } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| #include <spdlog/spdlog.h> | ||||
| #include <sys/eventfd.h> | ||||
| #include <fstream> | ||||
| #include <cassert> | ||||
| #include "util/format.hpp" | ||||
|  | ||||
|  | ||||
| @@ -278,8 +279,13 @@ auto waybar::modules::Network::update() -> void { | ||||
|       fmt::arg("bandwidthUpBits", pow_format(bandwidth_up * 8ull / interval_.count(), "b/s")), | ||||
|       fmt::arg("bandwidthDownOctets", pow_format(bandwidth_down / interval_.count(), "o/s")), | ||||
|       fmt::arg("bandwidthUpOctets", pow_format(bandwidth_up / interval_.count(), "o/s"))); | ||||
|   if (text != label_.get_label()) { | ||||
|   if (text.compare(label_.get_label()) != 0) { | ||||
|     label_.set_markup(text); | ||||
|     if (text.empty()) { | ||||
|       event_box_.hide(); | ||||
|     } else { | ||||
|       event_box_.show(); | ||||
|     } | ||||
|   } | ||||
|   if (tooltipEnabled()) { | ||||
|     if (tooltip_format.empty() && config_["tooltip-format"].isString()) { | ||||
| @@ -437,7 +443,6 @@ out: | ||||
| } | ||||
|  | ||||
| void waybar::modules::Network::getInterfaceAddress() { | ||||
|   unsigned int    cidrRaw; | ||||
|   struct ifaddrs *ifaddr, *ifa; | ||||
|   cidr_ = 0; | ||||
|   int success = getifaddrs(&ifaddr); | ||||
| @@ -449,18 +454,34 @@ void waybar::modules::Network::getInterfaceAddress() { | ||||
|     if (ifa->ifa_addr != nullptr && ifa->ifa_addr->sa_family == family_ && | ||||
|         ifa->ifa_name == ifname_) { | ||||
|       char ipaddr[INET6_ADDRSTRLEN]; | ||||
|       ipaddr_ = inet_ntop(family_, | ||||
|                           &reinterpret_cast<struct sockaddr_in *>(ifa->ifa_addr)->sin_addr, | ||||
|                           ipaddr, | ||||
|                           INET6_ADDRSTRLEN); | ||||
|       char netmask[INET6_ADDRSTRLEN]; | ||||
|       auto net_addr = reinterpret_cast<struct sockaddr_in *>(ifa->ifa_netmask); | ||||
|       netmask_ = inet_ntop(family_, &net_addr->sin_addr, netmask, INET6_ADDRSTRLEN); | ||||
|       cidrRaw = net_addr->sin_addr.s_addr; | ||||
|       unsigned int cidr = 0; | ||||
|       while (cidrRaw) { | ||||
|         cidr += cidrRaw & 1; | ||||
|         cidrRaw >>= 1; | ||||
|       if (family_ == AF_INET) { | ||||
|         ipaddr_ = inet_ntop(AF_INET, | ||||
|                             &reinterpret_cast<struct sockaddr_in *>(ifa->ifa_addr)->sin_addr, | ||||
|                             ipaddr, | ||||
|                             INET_ADDRSTRLEN); | ||||
|         auto net_addr = reinterpret_cast<struct sockaddr_in *>(ifa->ifa_netmask); | ||||
|         netmask_ = inet_ntop(AF_INET, &net_addr->sin_addr, netmask, INET_ADDRSTRLEN); | ||||
|         unsigned int cidrRaw = net_addr->sin_addr.s_addr; | ||||
|         while (cidrRaw) { | ||||
|           cidr += cidrRaw & 1; | ||||
|           cidrRaw >>= 1; | ||||
|         } | ||||
|       } else { | ||||
|         ipaddr_ = inet_ntop(AF_INET6, | ||||
|                             &reinterpret_cast<struct sockaddr_in6 *>(ifa->ifa_addr)->sin6_addr, | ||||
|                             ipaddr, | ||||
|                             INET6_ADDRSTRLEN); | ||||
|         auto net_addr = reinterpret_cast<struct sockaddr_in6 *>(ifa->ifa_netmask); | ||||
|         netmask_ = inet_ntop(AF_INET6, &net_addr->sin6_addr, netmask, INET6_ADDRSTRLEN); | ||||
|         for (size_t i = 0; i < sizeof(net_addr->sin6_addr.s6_addr); ++i) { | ||||
|           unsigned char cidrRaw = net_addr->sin6_addr.s6_addr[i]; | ||||
|           while (cidrRaw) { | ||||
|             cidr += cidrRaw & 1; | ||||
|             cidrRaw >>= 1; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       cidr_ = cidr; | ||||
|       break; | ||||
|   | ||||
| @@ -21,7 +21,7 @@ waybar::modules::Pulseaudio::Pulseaudio(const std::string &id, const Json::Value | ||||
|   if (context_ == nullptr) { | ||||
|     throw std::runtime_error("pa_context_new() failed."); | ||||
|   } | ||||
|   if (pa_context_connect(context_, nullptr, PA_CONTEXT_NOAUTOSPAWN, nullptr) < 0) { | ||||
|   if (pa_context_connect(context_, nullptr, PA_CONTEXT_NOFAIL, nullptr) < 0) { | ||||
|     auto err = | ||||
|         fmt::format("pa_context_connect() failed: {}", pa_strerror(pa_context_errno(context_))); | ||||
|     throw std::runtime_error(err); | ||||
| @@ -52,7 +52,8 @@ void waybar::modules::Pulseaudio::contextStateCb(pa_context *c, void *data) { | ||||
|       pa_context_set_subscribe_callback(c, subscribeCb, data); | ||||
|       pa_context_subscribe( | ||||
|           c, | ||||
|           static_cast<enum pa_subscription_mask>(static_cast<int>(PA_SUBSCRIPTION_MASK_SINK) | | ||||
|           static_cast<enum pa_subscription_mask>(static_cast<int>(PA_SUBSCRIPTION_MASK_SERVER) | | ||||
|                                                  static_cast<int>(PA_SUBSCRIPTION_MASK_SINK) | | ||||
|                                                  static_cast<int>(PA_SUBSCRIPTION_MASK_SOURCE)), | ||||
|           nullptr, | ||||
|           nullptr); | ||||
| @@ -109,7 +110,9 @@ void waybar::modules::Pulseaudio::subscribeCb(pa_context *                 conte | ||||
|   if (operation != PA_SUBSCRIPTION_EVENT_CHANGE) { | ||||
|     return; | ||||
|   } | ||||
|   if (facility == PA_SUBSCRIPTION_EVENT_SINK) { | ||||
|   if (facility == PA_SUBSCRIPTION_EVENT_SERVER) { | ||||
|     pa_context_get_server_info(context, serverInfoCb, data); | ||||
|   } else if (facility == PA_SUBSCRIPTION_EVENT_SINK) { | ||||
|     pa_context_get_sink_info_by_index(context, idx, sinkInfoCb, data); | ||||
|   } else if (facility == PA_SUBSCRIPTION_EVENT_SOURCE) { | ||||
|     pa_context_get_source_info_by_index(context, idx, sourceInfoCb, data); | ||||
| @@ -131,15 +134,15 @@ void waybar::modules::Pulseaudio::volumeModifyCb(pa_context *c, int success, voi | ||||
|  */ | ||||
| void waybar::modules::Pulseaudio::sourceInfoCb(pa_context * /*context*/, const pa_source_info *i, | ||||
|                                                int /*eol*/, void *data) { | ||||
|   if (i != nullptr) { | ||||
|     auto self = static_cast<waybar::modules::Pulseaudio *>(data); | ||||
|   auto pa = static_cast<waybar::modules::Pulseaudio *>(data); | ||||
|   if (i != nullptr && pa->default_source_name_ == i->name) { | ||||
|     auto source_volume = static_cast<float>(pa_cvolume_avg(&(i->volume))) / float{PA_VOLUME_NORM}; | ||||
|     self->source_volume_ = std::round(source_volume * 100.0F); | ||||
|     self->source_idx_ = i->index; | ||||
|     self->source_muted_ = i->mute != 0; | ||||
|     self->source_desc_ = i->description; | ||||
|     self->source_port_name_ = i->active_port != nullptr ? i->active_port->name : "Unknown"; | ||||
|     self->dp.emit(); | ||||
|     pa->source_volume_ = std::round(source_volume * 100.0F); | ||||
|     pa->source_idx_ = i->index; | ||||
|     pa->source_muted_ = i->mute != 0; | ||||
|     pa->source_desc_ = i->description; | ||||
|     pa->source_port_name_ = i->active_port != nullptr ? i->active_port->name : "Unknown"; | ||||
|     pa->dp.emit(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -147,9 +150,9 @@ void waybar::modules::Pulseaudio::sourceInfoCb(pa_context * /*context*/, const p | ||||
|  * Called when the requested sink information is ready. | ||||
|  */ | ||||
| void waybar::modules::Pulseaudio::sinkInfoCb(pa_context * /*context*/, const pa_sink_info *i, | ||||
|                                              int /*eol*/, void *                           data) { | ||||
|   if (i != nullptr) { | ||||
|     auto pa = static_cast<waybar::modules::Pulseaudio *>(data); | ||||
|                                              int /*eol*/, void *data) { | ||||
|   auto pa = static_cast<waybar::modules::Pulseaudio *>(data); | ||||
|   if (i != nullptr && pa->default_sink_name_ == i->name) { | ||||
|     pa->pa_volume_ = i->volume; | ||||
|     float volume = static_cast<float>(pa_cvolume_avg(&(pa->pa_volume_))) / float{PA_VOLUME_NORM}; | ||||
|     pa->sink_idx_ = i->index; | ||||
| @@ -158,6 +161,9 @@ void waybar::modules::Pulseaudio::sinkInfoCb(pa_context * /*context*/, const pa_ | ||||
|     pa->desc_ = i->description; | ||||
|     pa->monitor_ = i->monitor_source_name; | ||||
|     pa->port_name_ = i->active_port != nullptr ? i->active_port->name : "Unknown"; | ||||
|     if (auto ff = pa_proplist_gets(i->proplist, PA_PROP_DEVICE_FORM_FACTOR)) { | ||||
|       pa->form_factor_ = ff; | ||||
|     } | ||||
|     pa->dp.emit(); | ||||
|   } | ||||
| } | ||||
| @@ -168,16 +174,20 @@ void waybar::modules::Pulseaudio::sinkInfoCb(pa_context * /*context*/, const pa_ | ||||
|  */ | ||||
| void waybar::modules::Pulseaudio::serverInfoCb(pa_context *context, const pa_server_info *i, | ||||
|                                                void *data) { | ||||
|   auto pa = static_cast<waybar::modules::Pulseaudio *>(data); | ||||
|   pa->default_sink_name_ = i->default_sink_name; | ||||
|   pa->default_source_name_ = i->default_source_name; | ||||
|  | ||||
|   pa_context_get_sink_info_by_name(context, i->default_sink_name, sinkInfoCb, data); | ||||
|   pa_context_get_source_info_by_name(context, i->default_source_name, sourceInfoCb, data); | ||||
| } | ||||
|  | ||||
| static const std::array<std::string, 9> ports = { | ||||
|     "headphones", | ||||
|     "headphone", | ||||
|     "speaker", | ||||
|     "hdmi", | ||||
|     "headset", | ||||
|     "handsfree", | ||||
|     "hands-free", | ||||
|     "portable", | ||||
|     "car", | ||||
|     "hifi", | ||||
| @@ -185,7 +195,7 @@ static const std::array<std::string, 9> ports = { | ||||
| }; | ||||
|  | ||||
| const std::string waybar::modules::Pulseaudio::getPortIcon() const { | ||||
|   std::string nameLC = port_name_; | ||||
|   std::string nameLC = port_name_ + form_factor_; | ||||
|   std::transform(nameLC.begin(), nameLC.end(), nameLC.begin(), ::tolower); | ||||
|   for (auto const &port : ports) { | ||||
|     if (nameLC.find(port) != std::string::npos) { | ||||
| @@ -197,21 +207,27 @@ const std::string waybar::modules::Pulseaudio::getPortIcon() const { | ||||
|  | ||||
| auto waybar::modules::Pulseaudio::update() -> void { | ||||
|   auto format = format_; | ||||
|   std::string format_name = "format"; | ||||
|   if (monitor_.find("a2dp_sink") != std::string::npos) { | ||||
|     format_name = format_name + "-bluetooth"; | ||||
|     label_.get_style_context()->add_class("bluetooth"); | ||||
|   } else { | ||||
|     label_.get_style_context()->remove_class("bluetooth"); | ||||
|   if (!alt_) { | ||||
|     std::string format_name = "format"; | ||||
|     if (monitor_.find("a2dp_sink") != std::string::npos) { | ||||
|       format_name = format_name + "-bluetooth"; | ||||
|       label_.get_style_context()->add_class("bluetooth"); | ||||
|     } else { | ||||
|       label_.get_style_context()->remove_class("bluetooth"); | ||||
|     } | ||||
|     if (muted_) { | ||||
|       // Check muted bluetooth format exist, otherwise fallback to default muted format | ||||
|       if (format_name != "format" && !config_[format_name + "-muted"].isString()) { | ||||
|         format_name = "format"; | ||||
|       } | ||||
|       format_name = format_name + "-muted"; | ||||
|       label_.get_style_context()->add_class("muted"); | ||||
|     } else { | ||||
|       label_.get_style_context()->remove_class("muted"); | ||||
|     } | ||||
|     format = | ||||
|       config_[format_name].isString() ? config_[format_name].asString() : format; | ||||
|   } | ||||
|   if (muted_ ) { | ||||
|     format_name = format_name + "-muted"; | ||||
|     label_.get_style_context()->add_class("muted"); | ||||
|   } else { | ||||
|     label_.get_style_context()->remove_class("muted"); | ||||
|   } | ||||
|   format = | ||||
|     config_[format_name].isString() ? config_[format_name].asString() : format; | ||||
|   // TODO: find a better way to split source/sink | ||||
|   std::string format_source = "{volume}%"; | ||||
|   if (source_muted_ && config_["format-source-muted"].isString()) { | ||||
|   | ||||
| @@ -265,7 +265,11 @@ void Item::updateImage() { | ||||
|         if (pixbuf->gobj() != nullptr) { | ||||
|           // An icon specified by path and filename may be the wrong size for | ||||
|           // the tray | ||||
|           pixbuf = pixbuf->scale_simple(icon_size, icon_size, Gdk::InterpType::INTERP_BILINEAR); | ||||
|           // Keep the aspect ratio and scale to make the height equal to icon_size | ||||
|           // If people have non square icons, assume they want it to grow in width not height | ||||
|           int width = icon_size * pixbuf->get_width() / pixbuf->get_height(); | ||||
|  | ||||
|           pixbuf = pixbuf->scale_simple(width, icon_size, Gdk::InterpType::INTERP_BILINEAR); | ||||
|           image.set(pixbuf); | ||||
|         } | ||||
|       } else { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ namespace waybar::modules::SNI { | ||||
| Tray::Tray(const std::string& id, const Bar& bar, const Json::Value& config) | ||||
|     : AModule(config, "tray", id), | ||||
|       box_(bar.vertical ? Gtk::ORIENTATION_VERTICAL : Gtk::ORIENTATION_HORIZONTAL, 0), | ||||
|       watcher_(nb_hosts_), | ||||
|       watcher_(SNI::Watcher::getInstance()), | ||||
|       host_(nb_hosts_, config, std::bind(&Tray::onAdd, this, std::placeholders::_1), | ||||
|             std::bind(&Tray::onRemove, this, std::placeholders::_1)) { | ||||
|   spdlog::warn( | ||||
|   | ||||
| @@ -3,14 +3,13 @@ | ||||
|  | ||||
| using namespace waybar::modules::SNI; | ||||
|  | ||||
| Watcher::Watcher(std::size_t id) | ||||
| Watcher::Watcher() | ||||
|     : bus_name_id_(Gio::DBus::own_name(Gio::DBus::BusType::BUS_TYPE_SESSION, | ||||
|                                        "org.kde.StatusNotifierWatcher", | ||||
|                                        sigc::mem_fun(*this, &Watcher::busAcquired), | ||||
|                                        Gio::DBus::SlotNameAcquired(), Gio::DBus::SlotNameLost(), | ||||
|                                        Gio::DBus::BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT | | ||||
|                                            Gio::DBus::BUS_NAME_OWNER_FLAGS_REPLACE)), | ||||
|       watcher_id_(id), | ||||
|       watcher_(sn_watcher_skeleton_new()) {} | ||||
|  | ||||
| Watcher::~Watcher() { | ||||
| @@ -23,6 +22,7 @@ Watcher::~Watcher() { | ||||
|     g_slist_free_full(items_, gfWatchFree); | ||||
|     items_ = nullptr; | ||||
|   } | ||||
|   Gio::DBus::unown_name(bus_name_id_); | ||||
|   auto iface = G_DBUS_INTERFACE_SKELETON(watcher_); | ||||
|   g_dbus_interface_skeleton_unexport(iface); | ||||
| } | ||||
| @@ -34,7 +34,7 @@ void Watcher::busAcquired(const Glib::RefPtr<Gio::DBus::Connection>& conn, Glib: | ||||
|   if (error != nullptr) { | ||||
|     // Don't print an error when a watcher is already present | ||||
|     if (error->code != 2) { | ||||
|       spdlog::error("Watcher {}: {}", watcher_id_, error->message); | ||||
|       spdlog::error("Watcher: {}", error->message); | ||||
|     } | ||||
|     g_error_free(error); | ||||
|     return; | ||||
|   | ||||
| @@ -10,19 +10,23 @@ Ipc::Ipc() { | ||||
| } | ||||
|  | ||||
| Ipc::~Ipc() { | ||||
|   // To fail the IPC header | ||||
|   write(fd_, "close-sway-ipc", 14); | ||||
|   write(fd_event_, "close-sway-ipc", 14); | ||||
|   thread_.stop(); | ||||
|  | ||||
|   if (fd_ > 0) { | ||||
|     // To fail the IPC header | ||||
|     write(fd_, "close-sway-ipc", 14); | ||||
|     close(fd_); | ||||
|     fd_ = -1; | ||||
|   } | ||||
|   if (fd_event_ > 0) { | ||||
|     write(fd_event_, "close-sway-ipc", 14); | ||||
|     close(fd_event_); | ||||
|     fd_event_ = -1; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void Ipc::setWorker(std::function<void()>&& func) { thread_ = func; } | ||||
|  | ||||
| const std::string Ipc::getSocketPath() const { | ||||
|   const char* env = getenv("SWAYSOCK"); | ||||
|   if (env != nullptr) { | ||||
|   | ||||
| @@ -8,7 +8,13 @@ Mode::Mode(const std::string& id, const Json::Value& config) | ||||
|   ipc_.subscribe(R"(["mode"])"); | ||||
|   ipc_.signal_event.connect(sigc::mem_fun(*this, &Mode::onEvent)); | ||||
|   // Launch worker | ||||
|   worker(); | ||||
|   ipc_.setWorker([this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception& e) { | ||||
|       spdlog::error("Mode: {}", e.what()); | ||||
|     } | ||||
|   }); | ||||
|   dp.emit(); | ||||
| } | ||||
|  | ||||
| @@ -31,16 +37,6 @@ void Mode::onEvent(const struct Ipc::ipc_response& res) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| void Mode::worker() { | ||||
|   thread_ = [this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception& e) { | ||||
|       spdlog::error("Mode: {}", e.what()); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
| auto Mode::update() -> void { | ||||
|   if (mode_.empty()) { | ||||
|     event_box_.hide(); | ||||
|   | ||||
| @@ -11,7 +11,13 @@ Window::Window(const std::string& id, const Bar& bar, const Json::Value& config) | ||||
|   // Get Initial focused window | ||||
|   getTree(); | ||||
|   // Launch worker | ||||
|   worker(); | ||||
|   ipc_.setWorker([this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception& e) { | ||||
|       spdlog::error("Window: {}", e.what()); | ||||
|     } | ||||
|   }); | ||||
| } | ||||
|  | ||||
| void Window::onEvent(const struct Ipc::ipc_response& res) { getTree(); } | ||||
| @@ -28,16 +34,6 @@ void Window::onCmd(const struct Ipc::ipc_response& res) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| void Window::worker() { | ||||
|   thread_ = [this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception& e) { | ||||
|       spdlog::error("Window: {}", e.what()); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
| auto Window::update() -> void { | ||||
|   if (!old_app_id_.empty()) { | ||||
|     bar_.window.get_style_context()->remove_class(old_app_id_); | ||||
|   | ||||
| @@ -22,7 +22,13 @@ Workspaces::Workspaces(const std::string &id, const Bar &bar, const Json::Value | ||||
|     window.signal_scroll_event().connect(sigc::mem_fun(*this, &Workspaces::handleScroll)); | ||||
|   } | ||||
|   // Launch worker | ||||
|   worker(); | ||||
|   ipc_.setWorker([this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception &e) { | ||||
|       spdlog::error("Workspaces: {}", e.what()); | ||||
|     } | ||||
|   }); | ||||
| } | ||||
|  | ||||
| void Workspaces::onEvent(const struct Ipc::ipc_response &res) { | ||||
| @@ -102,16 +108,6 @@ void Workspaces::onCmd(const struct Ipc::ipc_response &res) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| void Workspaces::worker() { | ||||
|   thread_ = [this] { | ||||
|     try { | ||||
|       ipc_.handleEvent(); | ||||
|     } catch (const std::exception &e) { | ||||
|       spdlog::error("Workspaces: {}", e.what()); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
| bool Workspaces::filterButtons() { | ||||
|   bool needReorder = false; | ||||
|   for (auto it = buttons_.begin(); it != buttons_.end();) { | ||||
| @@ -161,12 +157,13 @@ auto Workspaces::update() -> void { | ||||
|     if (needReorder) { | ||||
|       box_.reorder_child(button, it - workspaces_.begin()); | ||||
|     } | ||||
|     std::string output = getIcon((*it)["name"].asString(), *it); | ||||
|     std::string output = (*it)["name"].asString(); | ||||
|     if (config_["format"].isString()) { | ||||
|       auto format = config_["format"].asString(); | ||||
|       output = fmt::format(format, | ||||
|                            fmt::arg("icon", output), | ||||
|                            fmt::arg("name", trimWorkspaceName((*it)["name"].asString())), | ||||
|                            fmt::arg("icon", getIcon(output, *it)), | ||||
|                            fmt::arg("value", output), | ||||
|                            fmt::arg("name", trimWorkspaceName(output)), | ||||
|                            fmt::arg("index", (*it)["num"].asString())); | ||||
|     } | ||||
|     if (!config_["disable-markup"].asBool()) { | ||||
| @@ -211,6 +208,8 @@ std::string Workspaces::getIcon(const std::string &name, const Json::Value &node | ||||
|       if (config_["format-icons"][key].isString() && node[key].asBool()) { | ||||
|         return config_["format-icons"][key].asString(); | ||||
|       } | ||||
|     } else if (config_["format_icons"]["persistent"].isString() && node["target_output"].isString()) { | ||||
|       return config_["format-icons"]["persistent"].asString(); | ||||
|     } else if (config_["format-icons"][key].isString()) { | ||||
|       return config_["format-icons"][key].asString(); | ||||
|     } | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| #include "modules/temperature.hpp" | ||||
| #include <filesystem> | ||||
|  | ||||
| waybar::modules::Temperature::Temperature(const std::string& id, const Json::Value& config) | ||||
|     : ALabel(config, "temperature", id, "{temperatureC}°C", 10) { | ||||
|   if (config_["hwmon-path"].isString()) { | ||||
|     file_path_ = config_["hwmon-path"].asString(); | ||||
|   } else if (config_["hwmon-path-abs"].isString() && config_["input-filename"].isString()) { | ||||
|     file_path_ = (*std::filesystem::directory_iterator(config_["hwmon-path-abs"].asString())).path().u8string() + "/" + config_["input-filename"].asString(); | ||||
|   } else { | ||||
|     auto zone = config_["thermal-zone"].isInt() ? config_["thermal-zone"].asInt() : 0; | ||||
|     file_path_ = fmt::format("/sys/class/thermal/thermal_zone{}/temp", zone); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Alex
					Alex