2018-08-08 23:54:58 +02:00
|
|
|
#include "modules/battery.hpp"
|
2020-05-22 21:23:04 +02:00
|
|
|
|
2019-05-19 01:44:45 +02:00
|
|
|
#include <spdlog/spdlog.h>
|
2018-08-08 23:54:58 +02:00
|
|
|
|
2018-12-18 17:30:54 +01:00
|
|
|
waybar::modules::Battery::Battery(const std::string& id, const Json::Value& config)
|
2019-05-22 12:06:24 +02:00
|
|
|
: ALabel(config, "battery", id, "{capacity}%", 60) {
|
2018-12-24 08:37:10 +01:00
|
|
|
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_) {
|
2018-12-24 08:37:10 +01:00
|
|
|
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() {
|
2018-12-24 08:37:10 +01:00
|
|
|
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] {
|
2018-10-25 17:39:15 +02:00
|
|
|
dp.emit();
|
2018-12-24 08:50:58 +01:00
|
|
|
thread_timer_.sleep_for(interval_);
|
2018-10-25 17:39:15 +02:00
|
|
|
};
|
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));
|
2018-12-24 08:37:10 +01:00
|
|
|
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
|
|
|
}
|
2018-11-09 23:02:46 +01: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() {
|
2018-12-24 08:37:10 +01:00
|
|
|
try {
|
2019-05-12 19:53:14 +02:00
|
|
|
for (auto& node : fs::directory_iterator(data_dir_)) {
|
2018-12-24 08:37:10 +01:00
|
|
|
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) &&
|
2019-05-12 19:53:14 +02:00
|
|
|
fs::exists(node.path() / "capacity") && fs::exists(node.path() / "uevent") &&
|
2020-06-16 12:30:21 +02:00
|
|
|
fs::exists(node.path() / "status") && fs::exists(node.path() / "type")) {
|
|
|
|
std::string type;
|
|
|
|
std::ifstream(node.path() / "type") >> type;
|
|
|
|
|
|
|
|
if (!type.compare("Battery")){
|
|
|
|
batteries_.push_back(node.path());
|
|
|
|
}
|
2018-12-24 08:37:10 +01:00
|
|
|
}
|
|
|
|
auto adap_defined = config_["adapter"].isString();
|
2019-04-18 17:52:00 +02:00
|
|
|
if (((adap_defined && dir_name == config_["adapter"].asString()) || !adap_defined) &&
|
2019-05-12 19:53:14 +02:00
|
|
|
fs::exists(node.path() / "online")) {
|
|
|
|
adapter_ = node.path();
|
2018-12-24 08:37:10 +01:00
|
|
|
}
|
|
|
|
}
|
2019-04-18 17:52:00 +02:00
|
|
|
} catch (fs::filesystem_error& e) {
|
2018-12-24 08:37:10 +01:00
|
|
|
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-21 19:36:14 +02:00
|
|
|
const std::tuple<uint8_t, float, std::string> waybar::modules::Battery::getInfos() const {
|
2018-08-08 23:54:58 +02:00
|
|
|
try {
|
2019-05-22 10:09:05 +02:00
|
|
|
uint32_t total_power = 0; // μW
|
|
|
|
uint32_t total_energy = 0; // μWh
|
2019-05-21 19:36:14 +02:00
|
|
|
uint32_t total_energy_full = 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-05-21 19:36:14 +02:00
|
|
|
uint32_t power_now;
|
|
|
|
uint32_t energy_full;
|
|
|
|
uint32_t energy_now;
|
2018-08-15 22:19:17 +02:00
|
|
|
std::string _status;
|
|
|
|
std::ifstream(bat / "status") >> _status;
|
2019-05-22 10:06:54 +02:00
|
|
|
auto rate_path = fs::exists(bat / "current_now") ? "current_now" : "power_now";
|
|
|
|
std::ifstream(bat / rate_path) >> power_now;
|
|
|
|
auto now_path = fs::exists(bat / "charge_now") ? "charge_now" : "energy_now";
|
|
|
|
std::ifstream(bat / now_path) >> energy_now;
|
|
|
|
auto full_path = fs::exists(bat / "charge_full") ? "charge_full" : "energy_full";
|
|
|
|
std::ifstream(bat / full_path) >> energy_full;
|
2018-08-16 14:29:41 +02:00
|
|
|
if (_status != "Unknown") {
|
2018-08-15 22:19:17 +02:00
|
|
|
status = _status;
|
2018-08-16 14:29:41 +02:00
|
|
|
}
|
2019-05-21 19:36:14 +02:00
|
|
|
total_power += power_now;
|
|
|
|
total_energy += energy_now;
|
|
|
|
total_energy_full += energy_full;
|
2018-08-08 23:54:58 +02:00
|
|
|
}
|
2019-05-17 10:59:54 +02:00
|
|
|
if (!adapter_.empty() && status == "Discharging") {
|
|
|
|
bool online;
|
|
|
|
std::ifstream(adapter_ / "online") >> online;
|
|
|
|
if (online) {
|
|
|
|
status = "Plugged";
|
|
|
|
}
|
2019-05-14 15:43:57 +02:00
|
|
|
}
|
2019-05-21 19:36:14 +02:00
|
|
|
float time_remaining = 0;
|
|
|
|
if (status == "Discharging" && total_power != 0) {
|
2019-05-22 10:09:05 +02:00
|
|
|
time_remaining = (float)total_energy / total_power;
|
2019-05-21 19:36:14 +02:00
|
|
|
} else if (status == "Charging" && total_power != 0) {
|
2019-05-22 10:09:05 +02:00
|
|
|
time_remaining = -(float)(total_energy_full - total_energy) / total_power;
|
2020-11-26 03:16:57 +01:00
|
|
|
if (time_remaining > 0.0f) {
|
|
|
|
// If we've turned positive it means the battery is past 100% and so
|
|
|
|
// just report that as no time remaining
|
|
|
|
time_remaining = 0.0f;
|
|
|
|
}
|
2019-05-21 19:36:14 +02:00
|
|
|
}
|
2020-11-26 02:02:06 +01:00
|
|
|
float capacity = ((float)total_energy * 100.0f / (float) total_energy_full);
|
2020-04-05 16:56:51 +02:00
|
|
|
// Handle full-at
|
|
|
|
if (config_["full-at"].isUInt()) {
|
|
|
|
auto full_at = config_["full-at"].asUInt();
|
|
|
|
if (full_at < 100) {
|
2020-04-21 00:53:36 +02:00
|
|
|
capacity = 100.f * capacity / full_at;
|
2020-04-05 16:56:51 +02:00
|
|
|
if (capacity > full_at) {
|
|
|
|
capacity = full_at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-11-26 03:16:57 +01:00
|
|
|
if (capacity > 100.f) {
|
|
|
|
// This can happen when the battery is calibrating and goes above 100%
|
|
|
|
// Handle it gracefully by clamping at 100% and presenting it as full
|
|
|
|
capacity = 100.f;
|
|
|
|
status = "Full";
|
|
|
|
}
|
2019-05-21 19:36:14 +02:00
|
|
|
return {capacity, time_remaining, status};
|
2018-11-02 11:23:29 +01:00
|
|
|
} catch (const std::exception& e) {
|
2019-05-22 10:09:05 +02:00
|
|
|
spdlog::error("Battery: {}", e.what());
|
2019-05-21 19:36:14 +02:00
|
|
|
return {0, 0, "Unknown"};
|
2018-11-02 11:23:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-22 10:09:05 +02:00
|
|
|
const std::string waybar::modules::Battery::getAdapterStatus(uint8_t capacity) const {
|
2018-12-24 08:37:10 +01:00
|
|
|
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) {
|
2019-10-03 10:37:37 +02:00
|
|
|
return "Plugged";
|
2019-05-14 15:43:57 +02:00
|
|
|
}
|
|
|
|
return "Discharging";
|
2018-12-24 08:37:10 +01:00
|
|
|
}
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
|
2019-05-21 19:36:14 +02:00
|
|
|
const std::string waybar::modules::Battery::formatTimeRemaining(float hoursRemaining) {
|
2019-05-22 10:09:05 +02:00
|
|
|
hoursRemaining = std::fabs(hoursRemaining);
|
|
|
|
uint16_t full_hours = static_cast<uint16_t>(hoursRemaining);
|
|
|
|
uint16_t minutes = static_cast<uint16_t>(60 * (hoursRemaining - full_hours));
|
2020-05-22 21:23:04 +02:00
|
|
|
auto format = std::string("{H} h {M} min");
|
2019-09-04 20:43:52 +02:00
|
|
|
if (config_["format-time"].isString()) {
|
|
|
|
format = config_["format-time"].asString();
|
|
|
|
}
|
|
|
|
return fmt::format(format, fmt::arg("H", full_hours), fmt::arg("M", minutes));
|
2019-05-21 19:36:14 +02:00
|
|
|
}
|
|
|
|
|
2019-04-18 17:52:00 +02:00
|
|
|
auto waybar::modules::Battery::update() -> void {
|
2019-05-21 19:36:14 +02:00
|
|
|
auto [capacity, time_remaining, status] = getInfos();
|
2018-12-24 12:17:07 +01:00
|
|
|
if (status == "Unknown") {
|
2019-05-21 19:35:39 +02:00
|
|
|
status = getAdapterStatus(capacity);
|
2018-12-24 12:17:07 +01:00
|
|
|
}
|
2019-02-22 11:35:26 +01:00
|
|
|
if (tooltipEnabled()) {
|
2019-05-21 19:36:14 +02:00
|
|
|
std::string tooltip_text;
|
|
|
|
if (time_remaining != 0) {
|
2019-05-22 10:09:05 +02:00
|
|
|
std::string time_to = std::string("Time to ") + ((time_remaining > 0) ? "empty" : "full");
|
|
|
|
tooltip_text = time_to + ": " + formatTimeRemaining(time_remaining);
|
2019-05-21 19:36:14 +02:00
|
|
|
} else {
|
2019-05-22 10:09:05 +02:00
|
|
|
tooltip_text = status;
|
2019-05-21 19:36:14 +02:00
|
|
|
}
|
|
|
|
label_.set_tooltip_text(tooltip_text);
|
2019-02-22 11:35:26 +01:00
|
|
|
}
|
2020-05-22 21:23:04 +02:00
|
|
|
// Transform to lowercase and replace space with dash
|
2019-12-30 13:55:49 +01:00
|
|
|
std::transform(status.begin(), status.end(), status.begin(), [](char ch) {
|
2020-05-22 21:23:04 +02:00
|
|
|
return ch == ' ' ? '-' : std::tolower(ch);
|
2019-12-30 13:55:49 +01:00
|
|
|
});
|
2018-11-02 11:23:29 +01:00
|
|
|
auto format = format_;
|
2019-05-13 11:36:34 +02:00
|
|
|
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_);
|
|
|
|
}
|
2018-11-02 22:04:43 +01:00
|
|
|
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()) {
|
2018-11-02 22:04:43 +01:00
|
|
|
format = config_["format-" + status].asString();
|
2018-11-02 22:50:01 +01:00
|
|
|
} else if (!state.empty() && config_["format-" + state].isString()) {
|
2018-11-02 11:23:29 +01:00
|
|
|
format = config_["format-" + state].asString();
|
|
|
|
}
|
|
|
|
if (format.empty()) {
|
|
|
|
event_box_.hide();
|
|
|
|
} else {
|
|
|
|
event_box_.show();
|
2020-05-22 21:23:04 +02:00
|
|
|
auto icons = std::vector<std::string>{status + "-" + state, status, state};
|
2019-05-22 10:09:05 +02:00
|
|
|
label_.set_markup(fmt::format(format,
|
|
|
|
fmt::arg("capacity", capacity),
|
2020-05-22 21:23:04 +02:00
|
|
|
fmt::arg("icon", getIcon(capacity, icons)),
|
2019-05-22 10:09:05 +02:00
|
|
|
fmt::arg("time", formatTimeRemaining(time_remaining))));
|
2018-08-08 23:54:58 +02:00
|
|
|
}
|
2020-04-12 18:30:21 +02:00
|
|
|
// Call parent update
|
|
|
|
ALabel::update();
|
2018-08-08 23:54:58 +02:00
|
|
|
}
|