2021-02-07 21:05:11 +01:00
|
|
|
#include "modules/keyboard_state.hpp"
|
2022-04-06 08:37:19 +02:00
|
|
|
|
2022-02-25 18:22:55 +01:00
|
|
|
#include <errno.h>
|
2021-02-08 03:05:34 +01:00
|
|
|
#include <spdlog/spdlog.h>
|
2022-02-25 18:22:55 +01:00
|
|
|
#include <string.h>
|
2021-02-07 21:05:11 +01:00
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
#include <filesystem>
|
|
|
|
|
2021-02-07 21:05:11 +01:00
|
|
|
extern "C" {
|
|
|
|
#include <fcntl.h>
|
2022-08-22 14:36:21 +02:00
|
|
|
#include <libinput.h>
|
|
|
|
#include <linux/input-event-codes.h>
|
|
|
|
#include <poll.h>
|
2022-04-06 08:37:19 +02:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
2022-08-22 14:36:21 +02:00
|
|
|
#include <unistd.h>
|
2021-02-07 21:05:11 +01:00
|
|
|
}
|
|
|
|
|
2022-02-25 18:22:55 +01:00
|
|
|
class errno_error : public std::runtime_error {
|
|
|
|
public:
|
|
|
|
int code;
|
|
|
|
errno_error(int code, const std::string& msg)
|
2022-04-06 08:37:19 +02: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 18:22:55 +01:00
|
|
|
private:
|
|
|
|
static auto getErrorMsg(int err, const char* msg) -> std::string {
|
|
|
|
std::string error_msg{msg};
|
|
|
|
error_msg += ": ";
|
2022-02-25 18:56:22 +01:00
|
|
|
|
2022-02-25 19:28:47 +01:00
|
|
|
#if (__GLIBC__ >= 2) && (__GLIBC_MINOR__ >= 32)
|
2022-04-06 08:37:19 +02:00
|
|
|
// strerrorname_np gets the error code's name; it's nice to have, but it's a recent GNU
|
|
|
|
// extension
|
2022-02-25 19:28:47 +01:00
|
|
|
const auto errno_name = strerrorname_np(err);
|
|
|
|
error_msg += errno_name;
|
|
|
|
error_msg += " ";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
const auto errno_str = strerror(err);
|
2022-02-25 18:22:55 +01:00
|
|
|
error_msg += errno_str;
|
2022-02-25 18:56:22 +01:00
|
|
|
|
2022-02-25 18:22:55 +01: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 08:37:19 +02: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 18:22:55 +01:00
|
|
|
}
|
|
|
|
|
2022-08-22 14:36:21 +02:00
|
|
|
auto waybar::modules::KeyboardState::findKeyboards() -> void {
|
|
|
|
if (config_["device-path"].isString()) {
|
|
|
|
std::string dev_path = config_["device-path"].asString();
|
|
|
|
libinput_devices_[dev_path] = nullptr;
|
|
|
|
fd_ = openFile(dev_path, O_NONBLOCK | O_CLOEXEC | O_RDONLY);
|
|
|
|
dev_ = openDevice(fd_);
|
|
|
|
} else {
|
|
|
|
DIR* dev_dir = opendir("/dev/input/by-path");
|
|
|
|
if (dev_dir == nullptr) {
|
|
|
|
throw errno_error(errno, "Failed to open /dev/input");
|
|
|
|
}
|
|
|
|
dirent* ep;
|
|
|
|
while ((ep = readdir(dev_dir))) {
|
|
|
|
if (ep->d_type == DT_DIR) continue;
|
|
|
|
std::string dev_path = std::string("/dev/input/by-path/") + ep->d_name;
|
|
|
|
int fd = openFile(dev_path.c_str(), O_NONBLOCK | O_CLOEXEC | O_RDONLY);
|
|
|
|
try {
|
|
|
|
auto dev = openDevice(fd);
|
|
|
|
if (supportsLockStates(dev)) {
|
|
|
|
spdlog::info("Found device {} at '{}'", libevdev_get_name(dev), dev_path);
|
|
|
|
if (libinput_devices_.empty()) {
|
|
|
|
fd_ = fd;
|
|
|
|
dev_ = dev;
|
|
|
|
} else {
|
|
|
|
libevdev_free(dev);
|
|
|
|
closeFile(fd);
|
|
|
|
}
|
|
|
|
libinput_devices_[dev_path] = libinput_path_add_device(libinput_, dev_path.c_str());
|
|
|
|
} else {
|
|
|
|
libevdev_free(dev);
|
|
|
|
closeFile(fd);
|
|
|
|
}
|
|
|
|
} 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());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (dev_ == nullptr) {
|
|
|
|
throw errno_error(errno, "Failed to find keyboard device");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
waybar::modules::KeyboardState::KeyboardState(const std::string& id, const Bar& bar,
|
|
|
|
const Json::Value& config)
|
2021-07-23 15:45:07 +02:00
|
|
|
: AModule(config, "keyboard-state", id, false, !config["disable-scroll"].asBool()),
|
2021-02-08 00:46:39 +01: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 08:37:19 +02:00
|
|
|
: config_["format"]["numlock"].isString()
|
|
|
|
? config_["format"]["numlock"].asString()
|
|
|
|
: "{name} {icon}"),
|
2021-02-08 00:46:39 +01:00
|
|
|
capslock_format_(config_["format"].isString() ? config_["format"].asString()
|
2022-04-06 08:37:19 +02:00
|
|
|
: config_["format"]["capslock"].isString()
|
|
|
|
? config_["format"]["capslock"].asString()
|
|
|
|
: "{name} {icon}"),
|
2021-02-08 00:46:39 +01:00
|
|
|
scrolllock_format_(config_["format"].isString() ? config_["format"].asString()
|
2022-04-06 08:37:19 +02:00
|
|
|
: config_["format"]["scrolllock"].isString()
|
|
|
|
? config_["format"]["scrolllock"].asString()
|
|
|
|
: "{name} {icon}"),
|
|
|
|
interval_(
|
|
|
|
std::chrono::seconds(config_["interval"].isUInt() ? config_["interval"].asUInt() : 1)),
|
2021-02-08 00:46:39 +01:00
|
|
|
icon_locked_(config_["format-icons"]["locked"].isString()
|
2022-04-06 08:37:19 +02:00
|
|
|
? config_["format-icons"]["locked"].asString()
|
|
|
|
: "locked"),
|
2021-02-08 00:46:39 +01:00
|
|
|
icon_unlocked_(config_["format-icons"]["unlocked"].isString()
|
2022-04-06 08:37:19 +02:00
|
|
|
? config_["format-icons"]["unlocked"].asString()
|
|
|
|
: "unlocked"),
|
2021-02-08 03:05:34 +01:00
|
|
|
fd_(0),
|
2022-08-22 14:36:21 +02:00
|
|
|
dev_(nullptr),
|
|
|
|
libinput_(nullptr),
|
|
|
|
libinput_devices_({}) {
|
|
|
|
struct libinput_interface interface = {
|
|
|
|
[](const char* path, int flags, void* user_data) { return open(path, flags); },
|
|
|
|
[](int fd, void* user_data) { close(fd); }};
|
|
|
|
libinput_ = libinput_path_create_context(&interface, NULL);
|
|
|
|
|
2021-07-23 15:45:07 +02:00
|
|
|
box_.set_name("keyboard-state");
|
2021-02-08 00:46:39 +01:00
|
|
|
if (config_["numlock"].asBool()) {
|
2022-05-11 08:44:57 +02:00
|
|
|
numlock_label_.get_style_context()->add_class("numlock");
|
2021-02-08 00:46:39 +01:00
|
|
|
box_.pack_end(numlock_label_, false, false, 0);
|
|
|
|
}
|
|
|
|
if (config_["capslock"].asBool()) {
|
2022-05-11 08:44:57 +02:00
|
|
|
capslock_label_.get_style_context()->add_class("capslock");
|
2021-02-08 00:46:39 +01:00
|
|
|
box_.pack_end(capslock_label_, false, false, 0);
|
|
|
|
}
|
|
|
|
if (config_["scrolllock"].asBool()) {
|
2022-05-11 08:44:57 +02:00
|
|
|
scrolllock_label_.get_style_context()->add_class("scrolllock");
|
2021-02-08 00:46:39 +01:00
|
|
|
box_.pack_end(scrolllock_label_, false, false, 0);
|
|
|
|
}
|
|
|
|
if (!id.empty()) {
|
|
|
|
box_.get_style_context()->add_class(id);
|
|
|
|
}
|
|
|
|
event_box_.add(box_);
|
|
|
|
|
2022-08-22 14:36:21 +02:00
|
|
|
findKeyboards();
|
2021-02-07 21:05:11 +01:00
|
|
|
|
|
|
|
thread_ = [this] {
|
|
|
|
dp.emit();
|
2022-08-22 14:36:21 +02:00
|
|
|
while (1) {
|
|
|
|
struct pollfd fd = {libinput_get_fd(libinput_), POLLIN, 0};
|
|
|
|
poll(&fd, 1, -1);
|
|
|
|
libinput_dispatch(libinput_);
|
|
|
|
struct libinput_event* event;
|
|
|
|
while ((event = libinput_get_event(libinput_))) {
|
|
|
|
auto type = libinput_event_get_type(event);
|
|
|
|
if (type == LIBINPUT_EVENT_KEYBOARD_KEY) {
|
|
|
|
auto keyboard_event = libinput_event_get_keyboard_event(event);
|
|
|
|
auto state = libinput_event_keyboard_get_key_state(keyboard_event);
|
|
|
|
if (state == LIBINPUT_KEY_STATE_RELEASED) {
|
|
|
|
uint32_t key = libinput_event_keyboard_get_key(keyboard_event);
|
|
|
|
switch (key) {
|
|
|
|
case KEY_CAPSLOCK:
|
|
|
|
case KEY_NUMLOCK:
|
|
|
|
case KEY_SCROLLLOCK:
|
|
|
|
dp.emit();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (type == LIBINPUT_EVENT_DEVICE_REMOVED) {
|
|
|
|
// TODO: Handle device removal.
|
|
|
|
// Clear libinput_devices_ and re-find keyboards.
|
|
|
|
}
|
|
|
|
libinput_event_destroy(event);
|
|
|
|
}
|
|
|
|
}
|
2021-02-07 21:05:11 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
waybar::modules::KeyboardState::~KeyboardState() {
|
|
|
|
libevdev_free(dev_);
|
2022-02-25 18:22:55 +01:00
|
|
|
try {
|
|
|
|
closeFile(fd_);
|
|
|
|
} catch (const std::runtime_error& e) {
|
|
|
|
spdlog::warn(e.what());
|
2021-02-07 21:05:11 +01:00
|
|
|
}
|
2022-08-22 14:36:21 +02:00
|
|
|
for (const auto& [_, dev_ptr] : libinput_devices_) {
|
|
|
|
libinput_path_remove_device(dev_ptr);
|
|
|
|
}
|
2021-02-07 21:05:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
auto waybar::modules::KeyboardState::update() -> void {
|
2022-08-22 14:36:21 +02:00
|
|
|
sleep(0); // wait for keyboard status change
|
|
|
|
|
2021-02-07 21:05:11 +01:00
|
|
|
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 18:22:55 +01:00
|
|
|
if (-err != EAGAIN) {
|
|
|
|
throw errno_error(-err, "Failed to sync evdev device");
|
2021-02-07 21:05:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int numl = libevdev_get_event_value(dev_, EV_LED, LED_NUML);
|
2021-02-08 00:46:39 +01: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 21:05:11 +01:00
|
|
|
|
2021-02-08 00:57:12 +01:00
|
|
|
struct {
|
|
|
|
bool state;
|
|
|
|
Gtk::Label& label;
|
|
|
|
const std::string& format;
|
|
|
|
const char* name;
|
|
|
|
} label_states[] = {
|
2022-04-06 08:37:19 +02:00
|
|
|
{(bool)numl, numlock_label_, numlock_format_, "Num"},
|
|
|
|
{(bool)capsl, capslock_label_, capslock_format_, "Caps"},
|
|
|
|
{(bool)scrolll, scrolllock_label_, scrolllock_format_, "Scroll"},
|
2021-02-08 00:57:12 +01: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 21:05:11 +01:00
|
|
|
|
2021-02-08 00:46:39 +01:00
|
|
|
AModule::update();
|
2021-02-07 21:05:11 +01:00
|
|
|
}
|