waybar/src/modules/battery.cpp

107 lines
3.2 KiB
C++
Raw Normal View History

2018-08-08 23:54:58 +02:00
#include "modules/battery.hpp"
2018-08-20 14:50:45 +02:00
waybar::modules::Battery::Battery(const Json::Value& config)
2018-08-27 01:36:25 +02:00
: ALabel(config, "{capacity}%")
2018-08-08 23:54:58 +02:00
{
try {
2018-10-26 09:27:16 +02:00
if (config_["bat"].isString()) {
2018-10-25 17:30:26 +02:00
auto dir = data_dir_ / config_["bat"].asString();
if (fs::is_directory(dir) && fs::exists(dir / "capacity")
&& fs::exists(dir / "status") && fs::exists(dir / "uevent")) {
batteries_.push_back(dir);
}
} else {
for (auto const& node : fs::directory_iterator(data_dir_)) {
if (fs::is_directory(node) && fs::exists(node / "capacity")
&& fs::exists(node / "status") && fs::exists(node / "uevent")) {
batteries_.push_back(node);
}
2018-08-16 14:29:41 +02:00
}
2018-08-08 23:54:58 +02:00
}
} catch (fs::filesystem_error &e) {
2018-08-13 14:05:13 +02:00
throw std::runtime_error(e.what());
2018-08-08 23:54:58 +02:00
}
2018-08-16 14:29:41 +02:00
if (batteries_.empty()) {
2018-10-26 09:27:16 +02:00
if (config_["bat"].isString()) {
2018-10-25 17:30:26 +02:00
throw std::runtime_error("No battery named " + config_["bat"].asString());
}
2018-08-13 14:05:13 +02:00
throw std::runtime_error("No batteries.");
2018-08-16 14:29:41 +02:00
}
2018-08-19 13:39:57 +02:00
fd_ = inotify_init1(IN_CLOEXEC);
if (fd_ == -1) {
2018-08-13 14:05:13 +02:00
throw std::runtime_error("Unable to listen batteries.");
2018-08-16 14:29:41 +02:00
}
2018-09-04 23:50:08 +02:00
for (auto const& bat : batteries_) {
2018-08-19 13:39:57 +02:00
inotify_add_watch(fd_, (bat / "uevent").c_str(), IN_ACCESS);
2018-08-16 14:29:41 +02:00
}
label_.set_name("battery");
2018-08-20 14:50:45 +02:00
worker();
}
waybar::modules::Battery::~Battery()
{
close(fd_);
}
void waybar::modules::Battery::worker()
{
// Trigger first values
update();
2018-10-26 09:27:16 +02:00
uint32_t interval = config_["interval"].isUInt() ? config_["interval"].asUInt() : 60;
threadTimer_ = [this, interval] {
thread_.sleep_for(chrono::seconds(interval));
dp.emit();
};
2018-08-19 13:39:57 +02:00
thread_ = [this] {
2018-08-17 14:24:00 +02:00
struct inotify_event event = {0};
2018-08-19 13:39:57 +02:00
int nbytes = read(fd_, &event, sizeof(event));
2018-08-16 14:29:41 +02:00
if (nbytes != sizeof(event)) {
2018-08-13 14:05:13 +02:00
return;
2018-08-16 14:29:41 +02:00
}
threadTimer_.stop();
2018-08-20 14:50:45 +02:00
dp.emit();
2018-08-08 23:54:58 +02:00
};
}
auto waybar::modules::Battery::update() -> void
{
try {
2018-08-13 14:05:13 +02:00
uint16_t total = 0;
std::string status;
2018-09-04 23:50:08 +02:00
for (auto const& bat : batteries_) {
2018-08-13 14:05:13 +02:00
uint16_t capacity;
std::string _status;
2018-08-09 11:21:08 +02:00
std::ifstream(bat / "capacity") >> capacity;
std::ifstream(bat / "status") >> _status;
2018-08-16 14:29:41 +02:00
if (_status != "Unknown") {
status = _status;
2018-08-16 14:29:41 +02:00
}
2018-08-13 14:05:13 +02:00
total += capacity;
2018-08-08 23:54:58 +02:00
}
2018-08-16 14:29:41 +02:00
uint16_t capacity = total / batteries_.size();
2018-08-27 01:36:25 +02:00
label_.set_text(fmt::format(format_, fmt::arg("capacity", capacity),
2018-08-16 14:29:41 +02:00
fmt::arg("icon", getIcon(capacity))));
label_.set_tooltip_text(status);
bool charging = status == "Charging";
2018-08-16 14:29:41 +02:00
if (charging) {
label_.get_style_context()->add_class("charging");
} else {
label_.get_style_context()->remove_class("charging");
}
auto warning = config_["warning"].isUInt() ? config_["warning"].asUInt() : 30;
2018-10-26 09:27:16 +02:00
auto critical = config_["critical"].isUInt() ? config_["critical"].asUInt() : 15;
2018-08-16 14:29:41 +02:00
if (capacity <= critical && !charging) {
label_.get_style_context()->add_class("critical");
label_.get_style_context()->remove_class("warning");
} else if (capacity <= warning && !charging) {
2018-08-16 14:29:41 +02:00
label_.get_style_context()->add_class("warning");
label_.get_style_context()->remove_class("critical");
2018-08-16 14:29:41 +02:00
} else {
label_.get_style_context()->remove_class("critical");
2018-08-16 14:29:41 +02:00
label_.get_style_context()->remove_class("warning");
}
2018-08-13 14:05:13 +02:00
} catch (const std::exception& e) {
2018-08-08 23:54:58 +02:00
std::cerr << e.what() << std::endl;
}
}