Waybar/src/modules/keyboard_state.cpp

214 lines
7.0 KiB
C++
Raw Normal View History

2021-02-07 20:05:11 +00:00
#include "modules/keyboard_state.hpp"
2022-04-06 06:37:19 +00:00
2022-02-25 17:22:55 +00:00
#include <errno.h>
#include <spdlog/spdlog.h>
2022-02-25 17:22:55 +00:00
#include <string.h>
2021-02-07 20:05:11 +00:00
2022-04-06 06:37:19 +00:00
#include <filesystem>
2021-02-07 20:05:11 +00:00
extern "C" {
#include <fcntl.h>
2022-04-06 06:37:19 +00:00
#include <sys/stat.h>
#include <sys/types.h>
2021-02-07 20:05:11 +00:00
}
2022-02-25 17:22:55 +00:00
class errno_error : public std::runtime_error {
public:
int code;
errno_error(int code, const std::string& msg)
2022-04-06 06:37:19 +00:00
: std::runtime_error(getErrorMsg(code, msg.c_str())), code(code) {}
errno_error(int code, const char* msg) : std::runtime_error(getErrorMsg(code, msg)), code(code) {}
2022-02-25 17:22:55 +00:00
private:
static auto getErrorMsg(int err, const char* msg) -> std::string {
std::string error_msg{msg};
error_msg += ": ";
2022-02-25 17:56:22 +00:00
#if (__GLIBC__ >= 2) && (__GLIBC_MINOR__ >= 32)
2022-04-06 06:37:19 +00:00
// strerrorname_np gets the error code's name; it's nice to have, but it's a recent GNU
// extension
const auto errno_name = strerrorname_np(err);
error_msg += errno_name;
error_msg += " ";
#endif
const auto errno_str = strerror(err);
2022-02-25 17:22:55 +00:00
error_msg += errno_str;
2022-02-25 17:56:22 +00:00
2022-02-25 17:22:55 +00:00
return error_msg;
}
};
auto openFile(const std::string& path, int flags) -> int {
int fd = open(path.c_str(), flags);
if (fd < 0) {
if (errno == EACCES) {
throw errno_error(errno, "Can't open " + path + " (are you in the input group?)");
} else {
throw errno_error(errno, "Can't open " + path);
}
}
return fd;
}
auto closeFile(int fd) -> void {
int res = close(fd);
if (res < 0) {
throw errno_error(errno, "Can't close file");
}
}
auto openDevice(int fd) -> libevdev* {
libevdev* dev;
int err = libevdev_new_from_fd(fd, &dev);
if (err < 0) {
throw errno_error(-err, "Can't create libevdev device");
}
return dev;
}
auto supportsLockStates(const libevdev* dev) -> bool {
2022-04-06 06:37:19 +00:00
return libevdev_has_event_type(dev, EV_LED) && libevdev_has_event_code(dev, EV_LED, LED_NUML) &&
libevdev_has_event_code(dev, EV_LED, LED_CAPSL) &&
libevdev_has_event_code(dev, EV_LED, LED_SCROLLL);
2022-02-25 17:22:55 +00:00
}
2022-04-06 06:37:19 +00:00
waybar::modules::KeyboardState::KeyboardState(const std::string& id, const Bar& bar,
const Json::Value& config)
: AModule(config, "keyboard-state", id, false, !config["disable-scroll"].asBool()),
2021-02-07 23:46:39 +00:00
box_(bar.vertical ? Gtk::ORIENTATION_VERTICAL : Gtk::ORIENTATION_HORIZONTAL, 0),
numlock_label_(""),
capslock_label_(""),
numlock_format_(config_["format"].isString() ? config_["format"].asString()
2022-04-06 06:37:19 +00:00
: config_["format"]["numlock"].isString()
? config_["format"]["numlock"].asString()
: "{name} {icon}"),
2021-02-07 23:46:39 +00:00
capslock_format_(config_["format"].isString() ? config_["format"].asString()
2022-04-06 06:37:19 +00:00
: config_["format"]["capslock"].isString()
? config_["format"]["capslock"].asString()
: "{name} {icon}"),
2021-02-07 23:46:39 +00:00
scrolllock_format_(config_["format"].isString() ? config_["format"].asString()
2022-04-06 06:37:19 +00:00
: config_["format"]["scrolllock"].isString()
? config_["format"]["scrolllock"].asString()
: "{name} {icon}"),
interval_(
std::chrono::seconds(config_["interval"].isUInt() ? config_["interval"].asUInt() : 1)),
2021-02-07 23:46:39 +00:00
icon_locked_(config_["format-icons"]["locked"].isString()
2022-04-06 06:37:19 +00:00
? config_["format-icons"]["locked"].asString()
: "locked"),
2021-02-07 23:46:39 +00:00
icon_unlocked_(config_["format-icons"]["unlocked"].isString()
2022-04-06 06:37:19 +00:00
? config_["format-icons"]["unlocked"].asString()
: "unlocked"),
fd_(0),
dev_(nullptr) {
box_.set_name("keyboard-state");
2021-02-07 23:46:39 +00:00
if (config_["numlock"].asBool()) {
numlock_label_.get_style_context()->add_class("numlock");
2021-02-07 23:46:39 +00:00
box_.pack_end(numlock_label_, false, false, 0);
}
if (config_["capslock"].asBool()) {
capslock_label_.get_style_context()->add_class("capslock");
2021-02-07 23:46:39 +00:00
box_.pack_end(capslock_label_, false, false, 0);
}
if (config_["scrolllock"].asBool()) {
scrolllock_label_.get_style_context()->add_class("scrolllock");
2021-02-07 23:46:39 +00:00
box_.pack_end(scrolllock_label_, false, false, 0);
}
if (!id.empty()) {
box_.get_style_context()->add_class(id);
}
event_box_.add(box_);
2021-02-07 20:05:11 +00:00
if (config_["device-path"].isString()) {
std::string dev_path = config_["device-path"].asString();
2022-02-25 17:22:55 +00:00
fd_ = openFile(dev_path, O_NONBLOCK | O_CLOEXEC | O_RDONLY);
dev_ = openDevice(fd_);
2021-02-07 20:05:11 +00:00
} else {
DIR* dev_dir = opendir("/dev/input");
if (dev_dir == nullptr) {
2022-02-25 17:22:55 +00:00
throw errno_error(errno, "Failed to open /dev/input");
}
2022-04-06 06:37:19 +00:00
dirent* ep;
while ((ep = readdir(dev_dir))) {
if (ep->d_type != DT_CHR) continue;
std::string dev_path = std::string("/dev/input/") + ep->d_name;
2022-02-25 17:22:55 +00:00
int fd = openFile(dev_path.c_str(), O_NONBLOCK | O_CLOEXEC | O_RDONLY);
try {
2022-02-25 17:22:55 +00:00
auto dev = openDevice(fd);
if (supportsLockStates(dev)) {
2022-04-06 06:37:19 +00:00
spdlog::info("Found device {} at '{}'", libevdev_get_name(dev), dev_path);
2022-02-25 17:22:55 +00:00
fd_ = fd;
dev_ = dev;
break;
}
} catch (const errno_error& e) {
// ENOTTY just means the device isn't an evdev device, skip it
if (e.code != ENOTTY) {
spdlog::warn(e.what());
}
}
2022-02-25 17:22:55 +00:00
closeFile(fd);
}
if (dev_ == nullptr) {
2022-02-25 17:22:55 +00:00
throw errno_error(errno, "Failed to find keyboard device");
}
2021-02-07 20:05:11 +00:00
}
thread_ = [this] {
dp.emit();
thread_.sleep_for(interval_);
};
}
waybar::modules::KeyboardState::~KeyboardState() {
libevdev_free(dev_);
2022-02-25 17:22:55 +00:00
try {
closeFile(fd_);
} catch (const std::runtime_error& e) {
spdlog::warn(e.what());
2021-02-07 20:05:11 +00:00
}
}
auto waybar::modules::KeyboardState::update() -> void {
int err = LIBEVDEV_READ_STATUS_SUCCESS;
while (err == LIBEVDEV_READ_STATUS_SUCCESS) {
input_event ev;
err = libevdev_next_event(dev_, LIBEVDEV_READ_FLAG_NORMAL, &ev);
while (err == LIBEVDEV_READ_STATUS_SYNC) {
err = libevdev_next_event(dev_, LIBEVDEV_READ_FLAG_SYNC, &ev);
}
}
2022-02-25 17:22:55 +00:00
if (-err != EAGAIN) {
throw errno_error(-err, "Failed to sync evdev device");
2021-02-07 20:05:11 +00:00
}
int numl = libevdev_get_event_value(dev_, EV_LED, LED_NUML);
2021-02-07 23:46:39 +00:00
int capsl = libevdev_get_event_value(dev_, EV_LED, LED_CAPSL);
int scrolll = libevdev_get_event_value(dev_, EV_LED, LED_SCROLLL);
2021-02-07 20:05:11 +00:00
2021-02-07 23:57:12 +00:00
struct {
bool state;
Gtk::Label& label;
const std::string& format;
const char* name;
} label_states[] = {
2022-04-06 06:37:19 +00:00
{(bool)numl, numlock_label_, numlock_format_, "Num"},
{(bool)capsl, capslock_label_, capslock_format_, "Caps"},
{(bool)scrolll, scrolllock_label_, scrolllock_format_, "Scroll"},
2021-02-07 23:57:12 +00:00
};
for (auto& label_state : label_states) {
std::string text;
text = fmt::format(label_state.format,
fmt::arg("icon", label_state.state ? icon_locked_ : icon_unlocked_),
fmt::arg("name", label_state.name));
label_state.label.set_markup(text);
if (label_state.state) {
label_state.label.get_style_context()->add_class("locked");
} else {
label_state.label.get_style_context()->remove_class("locked");
}
}
2021-02-07 20:05:11 +00:00
2021-02-07 23:46:39 +00:00
AModule::update();
2021-02-07 20:05:11 +00:00
}