waybar/src/modules/battery.cpp

154 lines
4.7 KiB
C++
Raw Normal View History

2018-08-08 23:54:58 +02:00
#include "modules/battery.hpp"
2018-12-18 17:30:54 +01:00
waybar::modules::Battery::Battery(const std::string& id, const Json::Value& config)
2019-04-18 17:52:00 +02:00
: ALabel(config, "{capacity}%", 60) {
2018-12-18 17:30:54 +01:00
label_.set_name("battery");
if (!id.empty()) {
label_.get_style_context()->add_class(id);
}
getBatteries();
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_) {
auto wd = inotify_add_watch(fd_, (bat / "uevent").c_str(), IN_ACCESS);
if (wd != -1) {
wds_.push_back(wd);
}
2018-08-16 14:29:41 +02:00
}
2018-08-20 14:50:45 +02:00
worker();
}
2019-04-18 17:52:00 +02:00
waybar::modules::Battery::~Battery() {
for (auto wd : wds_) {
inotify_rm_watch(fd_, wd);
}
2018-08-20 14:50:45 +02:00
close(fd_);
}
2019-04-18 17:52:00 +02:00
void waybar::modules::Battery::worker() {
2018-11-23 11:57:37 +01:00
thread_timer_ = [this] {
dp.emit();
2018-12-24 08:50:58 +01:00
thread_timer_.sleep_for(interval_);
};
2018-08-19 13:39:57 +02:00
thread_ = [this] {
2018-08-17 14:24:00 +02:00
struct inotify_event event = {0};
2019-04-18 17:52:00 +02:00
int nbytes = read(fd_, &event, sizeof(event));
if (nbytes != sizeof(event) || event.mask & IN_IGNORED) {
thread_.stop();
2018-08-13 14:05:13 +02:00
return;
2018-08-16 14:29:41 +02:00
}
// TODO: don't stop timer for now since there is some bugs :?
2018-11-16 10:02:12 +01:00
// thread_timer_.stop();
2018-08-20 14:50:45 +02:00
dp.emit();
2018-08-08 23:54:58 +02:00
};
}
2019-04-18 17:52:00 +02:00
void waybar::modules::Battery::getBatteries() {
try {
for (auto& node : fs::directory_iterator(data_dir_)) {
if (!fs::is_directory(node)) {
continue;
}
auto dir_name = node.path().filename();
auto bat_defined = config_["bat"].isString();
2019-04-18 17:52:00 +02:00
if (((bat_defined && dir_name == config_["bat"].asString()) || !bat_defined) &&
fs::exists(node.path() / "capacity") && fs::exists(node.path() / "uevent") &&
fs::exists(node.path() / "status")) {
batteries_.push_back(node.path());
}
auto adap_defined = config_["adapter"].isString();
2019-04-18 17:52:00 +02:00
if (((adap_defined && dir_name == config_["adapter"].asString()) || !adap_defined) &&
fs::exists(node.path() / "online")) {
adapter_ = node.path();
}
}
2019-04-18 17:52:00 +02:00
} catch (fs::filesystem_error& e) {
throw std::runtime_error(e.what());
}
if (batteries_.empty()) {
if (config_["bat"].isString()) {
throw std::runtime_error("No battery named " + config_["bat"].asString());
}
throw std::runtime_error("No batteries.");
}
}
2019-05-14 15:43:57 +02:00
const std::tuple<uint8_t, uint32_t, std::string> waybar::modules::Battery::getInfos() const {
2018-08-08 23:54:58 +02:00
try {
2019-04-18 17:52:00 +02:00
uint16_t total = 0;
2019-05-14 15:43:57 +02:00
uint32_t total_current = 0;
2018-11-02 22:50:01 +01:00
std::string status = "Unknown";
2018-09-04 23:50:08 +02:00
for (auto const& bat : batteries_) {
2019-04-18 17:52:00 +02:00
uint16_t capacity;
2019-05-14 15:43:57 +02:00
uint32_t current_now;
std::string _status;
2018-08-09 11:21:08 +02:00
std::ifstream(bat / "capacity") >> capacity;
std::ifstream(bat / "status") >> _status;
2019-05-14 15:43:57 +02:00
std::ifstream(bat / "current_now") >> current_now;
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;
2019-05-14 15:43:57 +02:00
total_current += current_now;
2018-08-08 23:54:58 +02:00
}
2018-08-16 14:29:41 +02:00
uint16_t capacity = total / batteries_.size();
2019-05-14 15:43:57 +02:00
if (status == "Charging" && total_current != 0) {
status = "Plugged";
2019-05-14 15:43:57 +02:00
}
return {capacity, total_current, status};
} catch (const std::exception& e) {
std::cerr << e.what() << std::endl;
2019-05-14 15:43:57 +02:00
return {0, 0, "Unknown"};
}
}
2019-05-14 15:43:57 +02:00
const std::string waybar::modules::Battery::getAdapterStatus(uint8_t capacity,
uint32_t current_now) const {
if (!adapter_.empty()) {
bool online;
std::ifstream(adapter_ / "online") >> online;
if (capacity == 100) {
return "Full";
}
2019-05-14 15:43:57 +02:00
if (online) {
return current_now == 0 ? "Charging" : "Plugged";
}
return "Discharging";
}
return "Unknown";
}
2019-04-18 17:52:00 +02:00
auto waybar::modules::Battery::update() -> void {
2019-05-14 15:43:57 +02:00
auto [capacity, current_now, status] = getInfos();
2018-12-24 12:17:07 +01:00
if (status == "Unknown") {
2019-05-14 15:43:57 +02:00
status = getAdapterStatus(capacity, current_now);
2018-12-24 12:17:07 +01:00
}
2019-02-22 11:35:26 +01:00
if (tooltipEnabled()) {
label_.set_tooltip_text(status);
}
std::transform(status.begin(), status.end(), status.begin(), ::tolower);
auto format = format_;
auto state = getState(capacity, true);
2019-05-14 15:43:57 +02:00
if (!old_status_.empty()) {
label_.get_style_context()->remove_class(old_status_);
}
label_.get_style_context()->add_class(status);
old_status_ = status;
if (!state.empty() && config_["format-" + status + "-" + state].isString()) {
format = config_["format-" + status + "-" + state].asString();
2018-11-02 22:50:01 +01:00
} else if (config_["format-" + status].isString()) {
format = config_["format-" + status].asString();
2018-11-02 22:50:01 +01:00
} else if (!state.empty() && config_["format-" + state].isString()) {
format = config_["format-" + state].asString();
}
if (format.empty()) {
event_box_.hide();
} else {
event_box_.show();
2019-05-16 17:18:27 +02:00
label_.set_markup(fmt::format(
format, fmt::arg("capacity", capacity), fmt::arg("icon", getIcon(capacity, state))));
2018-08-08 23:54:58 +02:00
}
}