Global shortcuts using evdev
  • Rust 98.8%
  • Nix 1.2%
Find a file
2025-06-09 17:07:18 +02:00
.forgejo/workflows flake + ci rework 2025-06-09 17:07:18 +02:00
examples flake + ci rework 2025-06-09 17:07:18 +02:00
src flake + ci rework 2025-06-09 17:07:18 +02:00
.envrc update to streams 2023-06-17 15:52:26 +02:00
.gitignore update to streams 2023-06-17 15:52:26 +02:00
Cargo.lock flake + ci rework 2025-06-09 17:07:18 +02:00
Cargo.toml flake + ci rework 2025-06-09 17:07:18 +02:00
flake.lock flake + ci rework 2025-06-09 17:07:18 +02:00
flake.nix flake + ci rework 2025-06-09 17:07:18 +02:00
README.md flake + ci rework 2025-06-09 17:07:18 +02:00

evdev-shortcut

Global shortcuts using evdev

Usage

use std::path::PathBuf;
use glob::GlobError;
use evdev_shortcut::{ShortcutListener, Shortcut, Modifier, Key};
use tokio::pin;
use futures::stream::StreamExt;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let listener = ShortcutListener::new();
    listener.add(Shortcut::new(&[Modifier::Meta], Key::KeyN));

    let devices =
        glob::glob("/dev/input/by-id/*-kbd")?.collect::<Result<Vec<PathBuf>, GlobError>>()?;

    let stream = listener.listen(&devices)?;
    pin!(stream);

    while let Some(event) = stream.next().await {
        println!("{} {}", event.shortcut, event.state);
    }
    Ok(())
}

Note that raw access to evdev devices is a privileged operation and usually requires running with elevated privileges. See shortcutd for a solution to running the elevated input handling in a separate process.