diff --git a/.gitignore b/.gitignore index 2c6f84f..67d0d60 100644 --- a/.gitignore +++ b/.gitignore @@ -3,3 +3,4 @@ /Cargo.lock /.cargo/ /wmlog +/xinitrc.tmp diff --git a/Cargo.toml b/Cargo.toml index 3215225..5e213ec 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -7,9 +7,14 @@ edition = "2018" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -x11 = {version = "2.18.2", features = ["xlib"] } +x11 = {version = "2.18.2", features = ["xlib", "xft"] } log = "0.4.13" simple_logger = "1.11.0" dirs = "3.0.2" log4rs = "1.0.0" -indexmap = "1.6.2" \ No newline at end of file +indexmap = "1.6.2" +thiserror = "1.0.30" +bitflags = "1.3.2" +derivative = "2.2.0" +serde = { version = "1.0", features = ["derive"] } +toml = "0.5" \ No newline at end of file diff --git a/Makefile b/Makefile deleted file mode 100644 index e02bb7c..0000000 --- a/Makefile +++ /dev/null @@ -1,5 +0,0 @@ -run: - startx ./xinitrc :0 vt2 - -run-root: - sudo startx ./xinitrc :0 vt2 diff --git a/README.md b/README.md index 96d0821..8cdba05 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,5 @@ -# Unnamed WM +# No WM +## formerly Unnamed This Project is a x11 tiling window manager written in Rust and losely based on / inspired by suckless' [dwm](https://dwm.suckless.org/). @@ -11,5 +12,5 @@ Both `M-S-T` and `M-S-RET` will spawn an instance of `xterm`, `M-q` will kill th One big difference from dwm is the way I handle virtual screens, although this is mostly a placeholder mechanic that I will most likely change in the future. Currently I have 3 (or more) virtual screens in a list that can be rotated with `M-left` and `M-right`. -Unnamed WM also has optional gaps :^) -![Unnamed WM in a VM](/vm-ss.png) +No WM also has optional gaps :^) +![No WM in a VM](/vm-ss.png) diff --git a/abstract1.jpg b/abstract1.jpg deleted file mode 100644 index 88b9ad5..0000000 Binary files a/abstract1.jpg and /dev/null differ diff --git a/nowm.toml b/nowm.toml new file mode 100644 index 0000000..5a73047 --- /dev/null +++ b/nowm.toml @@ -0,0 +1,4 @@ +num_virtualscreens = 10 +mod_key = "Super" +gap = 5 +kill_clients_on_exit = false \ No newline at end of file diff --git a/runner.sh b/runner.sh new file mode 100755 index 0000000..78dae00 --- /dev/null +++ b/runner.sh @@ -0,0 +1,12 @@ +#!/bin/sh + +# binary supplied by cargo +bin=$1 + +# write temporary xinitrc +printf "exec $bin\n" > xinitrc.tmp + +# call xinit +XEPHYR_BIN=$(which Xephyr) + +[ -z "$XEPHYR_BIN" ] || exec xinit ./xinitrc.tmp -- $XEPHYR_BIN :100 -ac -screen 800x600 diff --git a/src/backends/keycodes.rs b/src/backends/keycodes.rs new file mode 100644 index 0000000..167c941 --- /dev/null +++ b/src/backends/keycodes.rs @@ -0,0 +1,211 @@ +#![allow(dead_code)] + +#[derive(Debug, Hash, Ord, PartialOrd, PartialEq, Eq, Clone, Copy)] +pub enum KeyOrButton { + Key(VirtualKeyCode), + Button(MouseButton), +} + +#[derive(Debug, Hash, Ord, PartialOrd, PartialEq, Eq, Clone, Copy)] +pub enum MouseButton { + Left, + Middle, + Right, + ScrollUp, + ScrollDown, + ScrollLeft, + ScrollRight, + Forward, + Backward, +} + +/// from winit +#[derive(Debug, Hash, Ord, PartialOrd, PartialEq, Eq, Clone, Copy)] +#[repr(u32)] +pub enum VirtualKeyCode { + One, + Two, + Three, + Four, + Five, + Six, + Seven, + Eight, + Nine, + Zero, + A, + B, + C, + D, + E, + F, + G, + H, + I, + J, + K, + L, + M, + N, + O, + P, + Q, + R, + S, + T, + U, + V, + W, + X, + Y, + Z, + + /// The Escape key, next to F1. + Escape, + + F1, + F2, + F3, + F4, + F5, + F6, + F7, + F8, + F9, + F10, + F11, + F12, + F13, + F14, + F15, + F16, + F17, + F18, + F19, + F20, + F21, + F22, + F23, + F24, + + /// Print Screen/SysRq. + Print, + /// Scroll Lock. + Scroll, + /// Pause/Break key, next to Scroll lock. + Pause, + + /// `Insert`, next to Backspace. + Insert, + Home, + Delete, + End, + PageDown, + PageUp, + + Left, + Up, + Right, + Down, + + /// The Backspace key, right over Enter. + // TODO: rename + Back, + /// The Enter key. + Return, + /// The space bar. + Space, + + /// The "Compose" key on Linux. + Compose, + + Caret, + + Numlock, + Numpad0, + Numpad1, + Numpad2, + Numpad3, + Numpad4, + Numpad5, + Numpad6, + Numpad7, + Numpad8, + Numpad9, + NumpadAdd, + NumpadDivide, + NumpadDecimal, + NumpadComma, + NumpadEnter, + NumpadEquals, + NumpadMultiply, + NumpadSubtract, + + AbntC1, + AbntC2, + Apostrophe, + Apps, + Asterisk, + At, + Ax, + Backslash, + Calculator, + Capital, + Colon, + Comma, + Convert, + Equals, + Grave, + Kana, + Kanji, + LAlt, + LBracket, + LControl, + LShift, + LWin, + Mail, + MediaSelect, + MediaStop, + Minus, + Mute, + MyComputer, + // also called "Next" + NavigateForward, + // also called "Prior" + NavigateBackward, + NextTrack, + NoConvert, + OEM102, + Period, + PlayPause, + Plus, + Power, + PrevTrack, + RAlt, + RBracket, + RControl, + RShift, + RWin, + Semicolon, + Slash, + Sleep, + Stop, + Sysrq, + Tab, + Underline, + Unlabeled, + VolumeDown, + VolumeUp, + Wake, + WebBack, + WebFavorites, + WebForward, + WebHome, + WebRefresh, + WebSearch, + WebStop, + Yen, + Copy, + Paste, + Cut, +} diff --git a/src/backends/mod.rs b/src/backends/mod.rs new file mode 100644 index 0000000..73f4ef0 --- /dev/null +++ b/src/backends/mod.rs @@ -0,0 +1,6 @@ +pub mod keycodes; +pub mod traits; +pub mod window_event; +pub mod xlib; + +pub use traits::*; diff --git a/src/backends/traits.rs b/src/backends/traits.rs new file mode 100644 index 0000000..603a3b8 --- /dev/null +++ b/src/backends/traits.rs @@ -0,0 +1,53 @@ +use super::{ + window_event, + window_event::{KeyOrMouseBind, Point}, +}; + +pub trait WindowServerBackend { + type Window; + //type WindowEvent = super::window_event::WindowEvent; + + fn build() -> Self; + + fn next_event(&mut self) -> window_event::WindowEvent; + fn handle_event(&mut self, event: window_event::WindowEvent); + + /// adds a keybind to the specified `window`, or globally if `window` is `None`. + /// add global keybind + fn add_keybind(&mut self, keybind: KeyOrMouseBind); + fn remove_keybind(&mut self, keybind: &KeyOrMouseBind); + + fn focus_window(&self, window: Self::Window); + fn unfocus_window(&self, window: Self::Window); + fn raise_window(&self, window: Self::Window); + fn hide_window(&self, window: Self::Window); + fn kill_window(&self, window: Self::Window); + fn get_parent_window(&self, window: Self::Window) -> Option; + fn configure_window( + &self, + window: Self::Window, + new_size: Option>, + new_pos: Option>, + new_border: Option, + ); + + fn screen_size(&self) -> Point; + fn get_window_size(&self, window: Self::Window) -> Option>; + + fn grab_cursor(&self); + fn ungrab_cursor(&self); + fn move_cursor(&self, window: Option, position: Point); + + fn all_windows(&self) -> Option>; + + fn set_active_window_border_color(&mut self, color_name: &str); + fn set_inactive_window_border_color(&mut self, color_name: &str); + + fn resize_window(&self, window: Self::Window, new_size: Point) { + self.configure_window(window, Some(new_size), None, None); + } + + fn move_window(&self, window: Self::Window, new_pos: Point) { + self.configure_window(window, None, Some(new_pos), None); + } +} diff --git a/src/backends/window_event.rs b/src/backends/window_event.rs new file mode 100644 index 0000000..c99beba --- /dev/null +++ b/src/backends/window_event.rs @@ -0,0 +1,377 @@ +#![allow(dead_code)] + +use super::keycodes::{KeyOrButton, MouseButton, VirtualKeyCode}; +use bitflags::bitflags; + +#[derive(Debug)] +pub enum WindowEvent { + KeyEvent(KeyEvent), + ButtonEvent(ButtonEvent), + MotionEvent(MotionEvent), + MapRequestEvent(MapEvent), + MapEvent(MapEvent), + UnmapEvent(UnmapEvent), + CreateEvent(CreateEvent), + DestroyEvent(DestroyEvent), + EnterEvent(EnterEvent), + ConfigureEvent(ConfigureEvent), + FullscreenEvent(FullscreenEvent), //1 { window: Window, event: 1 }, +} + +#[derive(Debug, PartialEq, Eq, Copy, Clone)] +pub enum KeyState { + Pressed, + Released, +} + +#[derive( + Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Clone, Copy, serde::Deserialize, +)] +#[repr(u8)] +pub enum ModifierKey { + Shift, + ShiftLock, + Control, + Alt, + AltGr, + /// Windows key on most keyboards + Super, + NumLock, +} + +bitflags! { + pub struct ModifierState: u32 { + const SHIFT = 0x01; + const SHIFT_LOCK = 0x010; + const CONTROL = 0x0100; + const ALT = 0x01000; + const ALT_GR = 0x010000; + const SUPER = 0x0100000; + const NUM_LOCK = 0x01000000; + const IGNORE_LOCK = Self::CONTROL.bits | Self::ALT.bits | + Self::ALT_GR.bits | Self::SUPER.bits| Self::SHIFT.bits; + } +} + +impl From<[ModifierKey; N]> for ModifierState { + fn from(slice: [ModifierKey; N]) -> Self { + let mut state = ModifierState::empty(); + for ele in slice { + state.insert_mod(ele); + } + + state + } +} + +impl ModifierState { + pub fn eq_ignore_lock(&self, rhs: &Self) -> bool { + let mask = Self::IGNORE_LOCK; + *self & mask == *rhs & mask + } + + pub fn with_mod(mut self, modifier: ModifierKey) -> Self { + self.insert_mod(modifier); + self + } + + pub fn unset_mod(&mut self, modifier: ModifierKey) { + self.set_mod(modifier, false); + } + + pub fn set_mod(&mut self, modifier: ModifierKey, state: bool) { + self.set( + match modifier { + ModifierKey::Shift => Self::SHIFT, + ModifierKey::ShiftLock => Self::SHIFT_LOCK, + ModifierKey::Control => Self::CONTROL, + ModifierKey::Alt => Self::ALT, + ModifierKey::AltGr => Self::ALT_GR, + ModifierKey::Super => Self::SUPER, + ModifierKey::NumLock => Self::NUM_LOCK, + }, + state, + ); + } + + pub fn insert_mod(&mut self, modifier: ModifierKey) { + self.set_mod(modifier, true); + } +} + +impl Into for ModifierKey { + fn into(self) -> u8 { + self as u8 + } +} + +#[derive(Debug)] +pub struct KeyEvent { + pub window: Window, + pub state: KeyState, + pub keycode: VirtualKeyCode, + pub modifierstate: ModifierState, +} + +impl KeyEvent { + pub fn new( + window: Window, + state: KeyState, + keycode: VirtualKeyCode, + modifierstate: ModifierState, + ) -> Self { + Self { + window, + state, + keycode, + modifierstate, + } + } +} + +#[derive(Debug, Copy, Clone, PartialEq, PartialOrd)] +pub struct Point +where + I: Copy + Clone + PartialEq + PartialOrd, +{ + pub x: I, + pub y: I, +} +impl From<(I, I)> for Point +where + I: Copy + Clone + PartialEq + PartialOrd, +{ + fn from(value: (I, I)) -> Self { + Self::from_tuple(value) + } +} + +impl Point +where + I: Copy + Clone + PartialEq + PartialOrd, +{ + pub fn new(x: I, y: I) -> Self { + Self { x, y } + } + + pub fn from_tuple(tuple: (I, I)) -> Self { + Self { + x: tuple.0, + y: tuple.1, + } + } + + pub fn as_tuple(&self) -> (I, I) { + (self.x, self.y) + } +} + +#[derive(Debug)] +pub struct ButtonEvent { + pub window: Window, + pub state: KeyState, + pub keycode: MouseButton, + pub cursor_position: Point, + pub modifierstate: ModifierState, +} + +impl ButtonEvent { + pub fn new( + window: Window, + state: KeyState, + keycode: MouseButton, + cursor_position: Point, + modifierstate: ModifierState, + ) -> Self { + Self { + window, + state, + keycode, + cursor_position, + modifierstate, + } + } +} + +#[derive(Debug)] +pub struct MotionEvent { + pub position: Point, + pub window: Window, +} + +impl MotionEvent { + pub fn new(position: Point, window: Window) -> Self { + Self { position, window } + } +} + +#[derive(Debug)] +pub struct MapEvent { + pub window: Window, +} + +#[derive(Debug)] +pub struct UnmapEvent { + pub window: Window, +} + +#[derive(Debug)] +pub struct EnterEvent { + pub window: Window, +} + +#[derive(Debug)] +pub struct DestroyEvent { + pub window: Window, +} + +impl DestroyEvent { + pub fn new(window: Window) -> Self { + Self { window } + } +} + +#[derive(Debug)] +pub struct CreateEvent { + pub window: Window, + pub position: Point, + pub size: Point, +} + +impl CreateEvent { + pub fn new(window: Window, position: Point, size: Point) -> Self { + Self { + window, + position, + size, + } + } +} + +#[derive(Debug)] +pub struct ConfigureEvent { + pub window: Window, + pub position: Point, + pub size: Point, +} + +impl ConfigureEvent { + pub fn new(window: Window, position: Point, size: Point) -> Self { + Self { + window, + position, + size, + } + } +} + +#[derive(Debug)] +pub struct FullscreenEvent { + window: Window, + new_fullscreen: bool, +} + +impl FullscreenEvent { + pub fn new(window: Window, new_fullscreen: bool) -> Self { + Self { + window, + new_fullscreen, + } + } +} + +#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord)] +pub struct KeyBind { + pub key: VirtualKeyCode, + pub modifiers: ModifierState, +} + +impl KeyBind { + pub fn new(key: VirtualKeyCode) -> Self { + Self { + key, + modifiers: ModifierState::empty(), + } + } + + pub fn with_mod(mut self, modifier_key: ModifierKey) -> Self { + self.modifiers.insert_mod(modifier_key); + self + } +} + +#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord)] +pub struct MouseBind { + pub button: MouseButton, + pub modifiers: ModifierState, +} + +impl MouseBind { + pub fn new(button: MouseButton) -> Self { + Self { + button, + modifiers: ModifierState::empty(), + } + } + + pub fn with_mod(mut self, modifier_key: ModifierKey) -> Self { + self.modifiers.insert_mod(modifier_key); + self + } +} + +#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord)] +pub struct KeyOrMouseBind { + pub key: KeyOrButton, + pub modifiers: ModifierState, +} + +impl KeyOrMouseBind { + pub fn new(key: KeyOrButton) -> Self { + Self { + key, + modifiers: ModifierState::empty(), + } + } + + pub fn with_mod(mut self, modifier_key: ModifierKey) -> Self { + self.modifiers.insert_mod(modifier_key); + self + } +} + +impl From<&KeyBind> for KeyOrMouseBind { + fn from(keybind: &KeyBind) -> Self { + Self { + key: KeyOrButton::Key(keybind.key), + modifiers: keybind.modifiers, + } + } +} + +impl From for KeyOrMouseBind { + fn from(keybind: KeyBind) -> Self { + Self { + key: KeyOrButton::Key(keybind.key), + modifiers: keybind.modifiers, + } + } +} + +impl From<&MouseBind> for KeyOrMouseBind { + fn from(mousebind: &MouseBind) -> Self { + Self { + key: KeyOrButton::Button(mousebind.button), + modifiers: mousebind.modifiers, + } + } +} + +impl From for KeyOrMouseBind { + fn from(mousebind: MouseBind) -> Self { + Self { + key: KeyOrButton::Button(mousebind.button), + modifiers: mousebind.modifiers, + } + } +} diff --git a/src/backends/xlib/color.rs b/src/backends/xlib/color.rs new file mode 100644 index 0000000..f1e3cdf --- /dev/null +++ b/src/backends/xlib/color.rs @@ -0,0 +1,46 @@ +use std::mem::MaybeUninit; + +use x11::{xft, xlib}; + +use super::Display; + +pub struct XftColor { + inner: xft::XftColor, +} + +impl XftColor { + pub fn pixel(&self) -> u64 { + self.inner.pixel + } + + #[allow(dead_code)] + pub fn color(&self) -> x11::xrender::XRenderColor { + self.inner.color + } + + pub fn new( + dpy: Display, + screen: i32, + mut color_name: String, + ) -> Result { + color_name.push('\0'); + let mut color = MaybeUninit::::zeroed(); + + unsafe { + xft::XftColorAllocName( + dpy.get(), + xlib::XDefaultVisual(dpy.get(), screen), + xlib::XDefaultColormap(dpy.get(), screen), + color_name.as_ptr() as *mut _, + color.as_mut_ptr(), + ) != 0 + } + .then(|| Self { + inner: unsafe { color.assume_init() }, + }) + .ok_or(std::io::Error::new( + std::io::ErrorKind::NotFound, + "Unable to allocate color.", + )) + } +} diff --git a/src/backends/xlib/keysym.rs b/src/backends/xlib/keysym.rs new file mode 100644 index 0000000..1eaf08a --- /dev/null +++ b/src/backends/xlib/keysym.rs @@ -0,0 +1,2108 @@ +#![allow(unreachable_patterns)] +use std::{borrow::Borrow, ops::Deref}; + +use crate::backends::keycodes::{MouseButton, VirtualKeyCode}; + +pub fn xev_to_mouse_button( + button: &x11::xlib::XButtonEvent, +) -> Option { + match button.button { + 1 => Some(MouseButton::Left), + 2 => Some(MouseButton::Middle), + 3 => Some(MouseButton::Right), + 4 => Some(MouseButton::ScrollUp), + 5 => Some(MouseButton::ScrollDown), + 6 => Some(MouseButton::ScrollLeft), + 7 => Some(MouseButton::ScrollRight), + 8 => Some(MouseButton::Forward), + 9 => Some(MouseButton::Backward), + _ => None, + } +} + +pub fn mouse_button_to_xbutton(button: MouseButton) -> i32 { + match button { + MouseButton::Left => 1, + MouseButton::Middle => 2, + MouseButton::Right => 3, + MouseButton::ScrollUp => 4, + MouseButton::ScrollDown => 5, + MouseButton::ScrollLeft => 6, + MouseButton::ScrollRight => 7, + MouseButton::Forward => 8, + MouseButton::Backward => 9, + } +} + +#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone)] +pub struct XKeySym(pub u32); + +impl XKeySym { + pub fn new(value: u32) -> Self { + Self(value) + } + + pub fn get(&self) -> u32 { + self.0 + } +} + +impl Borrow for XKeySym { + fn borrow(&self) -> &u32 { + &self.0 + } +} + +impl AsRef for XKeySym { + fn as_ref(&self) -> &u32 { + &self.0 + } +} + +impl Deref for XKeySym { + type Target = u32; + + fn deref(&self) -> &Self::Target { + &self.0 + } +} + +impl From for VirtualKeyCode { + fn from(value: XKeySym) -> Self { + keysym_to_virtual_keycode(*value).unwrap() + } +} +impl From for XKeySym { + fn from(value: VirtualKeyCode) -> Self { + Self(virtual_keycode_to_keysym(value).unwrap()) + } +} + +/// from winit +pub fn keysym_to_virtual_keycode(keysym: u32) -> Option { + Some(match keysym { + x11::keysym::XK_BackSpace => VirtualKeyCode::Back, + x11::keysym::XK_Tab => VirtualKeyCode::Tab, + //x11::keysym::XK_Linefeed => VirtualKeyCode::Linefeed, + //x11::keysym::XK_Clear => VirtualKeyCode::Clear, + x11::keysym::XK_Return => VirtualKeyCode::Return, + //x11::keysym::XK_Pause => VirtualKeyCode::Pause, + //x11::keysym::XK_Scroll_Lock => VirtualKeyCode::Scroll_lock, + //x11::keysym::XK_Sys_Req => VirtualKeyCode::Sys_req, + x11::keysym::XK_Escape => VirtualKeyCode::Escape, + x11::keysym::XK_Delete => VirtualKeyCode::Delete, + x11::keysym::XK_Multi_key => VirtualKeyCode::Compose, + //x11::keysym::XK_Kanji => VirtualKeyCode::Kanji, + //x11::keysym::XK_Muhenkan => VirtualKeyCode::Muhenkan, + //x11::keysym::XK_Henkan_Mode => VirtualKeyCode::Henkan_mode, + //x11::keysym::XK_Henkan => VirtualKeyCode::Henkan, + //x11::keysym::XK_Romaji => VirtualKeyCode::Romaji, + //x11::keysym::XK_Hiragana => VirtualKeyCode::Hiragana, + //x11::keysym::XK_Katakana => VirtualKeyCode::Katakana, + //x11::keysym::XK_Hiragana_Katakana => VirtualKeyCode::Hiragana_katakana, + //x11::keysym::XK_Zenkaku => VirtualKeyCode::Zenkaku, + //x11::keysym::XK_Hankaku => VirtualKeyCode::Hankaku, + //x11::keysym::XK_Zenkaku_Hankaku => VirtualKeyCode::Zenkaku_hankaku, + //x11::keysym::XK_Touroku => VirtualKeyCode::Touroku, + //x11::keysym::XK_Massyo => VirtualKeyCode::Massyo, + //x11::keysym::XK_Kana_Lock => VirtualKeyCode::Kana_lock, + //x11::keysym::XK_Kana_Shift => VirtualKeyCode::Kana_shift, + //x11::keysym::XK_Eisu_Shift => VirtualKeyCode::Eisu_shift, + //x11::keysym::XK_Eisu_toggle => VirtualKeyCode::Eisu_toggle, + x11::keysym::XK_Home => VirtualKeyCode::Home, + x11::keysym::XK_Left => VirtualKeyCode::Left, + x11::keysym::XK_Up => VirtualKeyCode::Up, + x11::keysym::XK_Right => VirtualKeyCode::Right, + x11::keysym::XK_Down => VirtualKeyCode::Down, + //x11::keysym::XK_Prior => VirtualKeyCode::Prior, + x11::keysym::XK_Page_Up => VirtualKeyCode::PageUp, + //x11::keysym::XK_Next => VirtualKeyCode::Next, + x11::keysym::XK_Page_Down => VirtualKeyCode::PageDown, + x11::keysym::XK_End => VirtualKeyCode::End, + //x11::keysym::XK_Begin => VirtualKeyCode::Begin, + x11::keysym::XK_Win_L => VirtualKeyCode::LWin, + x11::keysym::XK_Win_R => VirtualKeyCode::RWin, + //x11::keysym::XK_App => VirtualKeyCode::App, + //x11::keysym::XK_Select => VirtualKeyCode::Select, + x11::keysym::XK_Print => VirtualKeyCode::Print, + //x11::keysym::XK_Execute => VirtualKeyCode::Execute, + x11::keysym::XK_Insert => VirtualKeyCode::Insert, + //x11::keysym::XK_Undo => VirtualKeyCode::Undo, + //x11::keysym::XK_Redo => VirtualKeyCode::Redo, + //x11::keysym::XK_Menu => VirtualKeyCode::Menu, + //x11::keysym::XK_Find => VirtualKeyCode::Find, + //x11::keysym::XK_Cancel => VirtualKeyCode::Cancel, + //x11::keysym::XK_Help => VirtualKeyCode::Help, + //x11::keysym::XK_Break => VirtualKeyCode::Break, + //x11::keysym::XK_Mode_switch => VirtualKeyCode::Mode_switch, + //x11::keysym::XK_script_switch => VirtualKeyCode::Script_switch, + //x11::keysym::XK_Num_Lock => VirtualKeyCode::Num_lock, + //x11::keysym::XK_KP_Space => VirtualKeyCode::Kp_space, + //x11::keysym::XK_KP_Tab => VirtualKeyCode::Kp_tab, + //x11::keysym::XK_KP_Enter => VirtualKeyCode::Kp_enter, + //x11::keysym::XK_KP_F1 => VirtualKeyCode::Kp_f1, + //x11::keysym::XK_KP_F2 => VirtualKeyCode::Kp_f2, + //x11::keysym::XK_KP_F3 => VirtualKeyCode::Kp_f3, + //x11::keysym::XK_KP_F4 => VirtualKeyCode::Kp_f4, + x11::keysym::XK_KP_Home => VirtualKeyCode::Home, + x11::keysym::XK_KP_Left => VirtualKeyCode::Left, + x11::keysym::XK_KP_Up => VirtualKeyCode::Up, + x11::keysym::XK_KP_Right => VirtualKeyCode::Right, + x11::keysym::XK_KP_Down => VirtualKeyCode::Down, + //x11::keysym::XK_KP_Prior => VirtualKeyCode::Kp_prior, + x11::keysym::XK_KP_Page_Up => VirtualKeyCode::PageUp, + //x11::keysym::XK_KP_Next => VirtualKeyCode::Kp_next, + x11::keysym::XK_KP_Page_Down => VirtualKeyCode::PageDown, + x11::keysym::XK_KP_End => VirtualKeyCode::End, + //x11::keysym::XK_KP_Begin => VirtualKeyCode::Kp_begin, + x11::keysym::XK_KP_Insert => VirtualKeyCode::Insert, + x11::keysym::XK_KP_Delete => VirtualKeyCode::Delete, + x11::keysym::XK_KP_Equal => VirtualKeyCode::NumpadEquals, + x11::keysym::XK_KP_Multiply => VirtualKeyCode::NumpadMultiply, + x11::keysym::XK_KP_Add => VirtualKeyCode::NumpadAdd, + //x11::keysym::XK_KP_Separator => VirtualKeyCode::Kp_separator, + x11::keysym::XK_KP_Subtract => VirtualKeyCode::NumpadSubtract, + x11::keysym::XK_KP_Decimal => VirtualKeyCode::NumpadDecimal, + x11::keysym::XK_KP_Divide => VirtualKeyCode::NumpadDivide, + x11::keysym::XK_KP_0 => VirtualKeyCode::Numpad0, + x11::keysym::XK_KP_1 => VirtualKeyCode::Numpad1, + x11::keysym::XK_KP_2 => VirtualKeyCode::Numpad2, + x11::keysym::XK_KP_3 => VirtualKeyCode::Numpad3, + x11::keysym::XK_KP_4 => VirtualKeyCode::Numpad4, + x11::keysym::XK_KP_5 => VirtualKeyCode::Numpad5, + x11::keysym::XK_KP_6 => VirtualKeyCode::Numpad6, + x11::keysym::XK_KP_7 => VirtualKeyCode::Numpad7, + x11::keysym::XK_KP_8 => VirtualKeyCode::Numpad8, + x11::keysym::XK_KP_9 => VirtualKeyCode::Numpad9, + x11::keysym::XK_F1 => VirtualKeyCode::F1, + x11::keysym::XK_F2 => VirtualKeyCode::F2, + x11::keysym::XK_F3 => VirtualKeyCode::F3, + x11::keysym::XK_F4 => VirtualKeyCode::F4, + x11::keysym::XK_F5 => VirtualKeyCode::F5, + x11::keysym::XK_F6 => VirtualKeyCode::F6, + x11::keysym::XK_F7 => VirtualKeyCode::F7, + x11::keysym::XK_F8 => VirtualKeyCode::F8, + x11::keysym::XK_F9 => VirtualKeyCode::F9, + x11::keysym::XK_F10 => VirtualKeyCode::F10, + x11::keysym::XK_F11 => VirtualKeyCode::F11, + //x11::keysym::XK_L1 => VirtualKeyCode::L1, + x11::keysym::XK_F12 => VirtualKeyCode::F12, + //x11::keysym::XK_L2 => VirtualKeyCode::L2, + x11::keysym::XK_F13 => VirtualKeyCode::F13, + //x11::keysym::XK_L3 => VirtualKeyCode::L3, + x11::keysym::XK_F14 => VirtualKeyCode::F14, + //x11::keysym::XK_L4 => VirtualKeyCode::L4, + x11::keysym::XK_F15 => VirtualKeyCode::F15, + //x11::keysym::XK_L5 => VirtualKeyCode::L5, + x11::keysym::XK_F16 => VirtualKeyCode::F16, + //x11::keysym::XK_L6 => VirtualKeyCode::L6, + x11::keysym::XK_F17 => VirtualKeyCode::F17, + //x11::keysym::XK_L7 => VirtualKeyCode::L7, + x11::keysym::XK_F18 => VirtualKeyCode::F18, + //x11::keysym::XK_L8 => VirtualKeyCode::L8, + x11::keysym::XK_F19 => VirtualKeyCode::F19, + //x11::keysym::XK_L9 => VirtualKeyCode::L9, + x11::keysym::XK_F20 => VirtualKeyCode::F20, + //x11::keysym::XK_L10 => VirtualKeyCode::L10, + x11::keysym::XK_F21 => VirtualKeyCode::F21, + //x11::keysym::XK_R1 => VirtualKeyCode::R1, + x11::keysym::XK_F22 => VirtualKeyCode::F22, + //x11::keysym::XK_R2 => VirtualKeyCode::R2, + x11::keysym::XK_F23 => VirtualKeyCode::F23, + //x11::keysym::XK_R3 => VirtualKeyCode::R3, + x11::keysym::XK_F24 => VirtualKeyCode::F24, + //x11::keysym::XK_R4 => VirtualKeyCode::R4, + //x11::keysym::XK_F25 => VirtualKeyCode::F25, + //x11::keysym::XK_R5 => VirtualKeyCode::R5, + //x11::keysym::XK_F26 => VirtualKeyCode::F26, + //x11::keysym::XK_R6 => VirtualKeyCode::R6, + //x11::keysym::XK_F27 => VirtualKeyCode::F27, + //x11::keysym::XK_R7 => VirtualKeyCode::R7, + //x11::keysym::XK_F28 => VirtualKeyCode::F28, + //x11::keysym::XK_R8 => VirtualKeyCode::R8, + //x11::keysym::XK_F29 => VirtualKeyCode::F29, + //x11::keysym::XK_R9 => VirtualKeyCode::R9, + //x11::keysym::XK_F30 => VirtualKeyCode::F30, + //x11::keysym::XK_R10 => VirtualKeyCode::R10, + //x11::keysym::XK_F31 => VirtualKeyCode::F31, + //x11::keysym::XK_R11 => VirtualKeyCode::R11, + //x11::keysym::XK_F32 => VirtualKeyCode::F32, + //x11::keysym::XK_R12 => VirtualKeyCode::R12, + //x11::keysym::XK_F33 => VirtualKeyCode::F33, + //x11::keysym::XK_R13 => VirtualKeyCode::R13, + //x11::keysym::XK_F34 => VirtualKeyCode::F34, + //x11::keysym::XK_R14 => VirtualKeyCode::R14, + //x11::keysym::XK_F35 => VirtualKeyCode::F35, + //x11::keysym::XK_R15 => VirtualKeyCode::R15, + x11::keysym::XK_Shift_L => VirtualKeyCode::LShift, + x11::keysym::XK_Shift_R => VirtualKeyCode::RShift, + x11::keysym::XK_Control_L => VirtualKeyCode::LControl, + x11::keysym::XK_Control_R => VirtualKeyCode::RControl, + //x11::keysym::XK_Caps_Lock => VirtualKeyCode::Caps_lock, + //x11::keysym::XK_Shift_Lock => VirtualKeyCode::Shift_lock, + //x11::keysym::XK_Meta_L => VirtualKeyCode::Meta_l, + //x11::keysym::XK_Meta_R => VirtualKeyCode::Meta_r, + x11::keysym::XK_Alt_L => VirtualKeyCode::LAlt, + x11::keysym::XK_Alt_R => VirtualKeyCode::RAlt, + //x11::keysym::XK_Super_L => VirtualKeyCode::Super_l, + //x11::keysym::XK_Super_R => VirtualKeyCode::Super_r, + //x11::keysym::XK_Hyper_L => VirtualKeyCode::Hyper_l, + //x11::keysym::XK_Hyper_R => VirtualKeyCode::Hyper_r, + x11::keysym::XK_ISO_Left_Tab => VirtualKeyCode::Tab, + x11::keysym::XK_space => VirtualKeyCode::Space, + //x11::keysym::XK_exclam => VirtualKeyCode::Exclam, + //x11::keysym::XK_quotedbl => VirtualKeyCode::Quotedbl, + //x11::keysym::XK_numbersign => VirtualKeyCode::Numbersign, + //x11::keysym::XK_dollar => VirtualKeyCode::Dollar, + //x11::keysym::XK_percent => VirtualKeyCode::Percent, + //x11::keysym::XK_ampersand => VirtualKeyCode::Ampersand, + x11::keysym::XK_apostrophe => VirtualKeyCode::Apostrophe, + //x11::keysym::XK_quoteright => VirtualKeyCode::Quoteright, + //x11::keysym::XK_parenleft => VirtualKeyCode::Parenleft, + //x11::keysym::XK_parenright => VirtualKeyCode::Parenright, + x11::keysym::XK_asterisk => VirtualKeyCode::Asterisk, + x11::keysym::XK_plus => VirtualKeyCode::Plus, + x11::keysym::XK_comma => VirtualKeyCode::Comma, + x11::keysym::XK_minus => VirtualKeyCode::Minus, + x11::keysym::XK_period => VirtualKeyCode::Period, + x11::keysym::XK_slash => VirtualKeyCode::Slash, + x11::keysym::XK_0 => VirtualKeyCode::Zero, + x11::keysym::XK_1 => VirtualKeyCode::One, + x11::keysym::XK_2 => VirtualKeyCode::Two, + x11::keysym::XK_3 => VirtualKeyCode::Three, + x11::keysym::XK_4 => VirtualKeyCode::Four, + x11::keysym::XK_5 => VirtualKeyCode::Five, + x11::keysym::XK_6 => VirtualKeyCode::Six, + x11::keysym::XK_7 => VirtualKeyCode::Seven, + x11::keysym::XK_8 => VirtualKeyCode::Eight, + x11::keysym::XK_9 => VirtualKeyCode::Nine, + x11::keysym::XK_colon => VirtualKeyCode::Colon, + x11::keysym::XK_semicolon => VirtualKeyCode::Semicolon, + //x11::keysym::XK_less => VirtualKeyCode::Less, + x11::keysym::XK_equal => VirtualKeyCode::Equals, + //x11::keysym::XK_greater => VirtualKeyCode::Greater, + //x11::keysym::XK_question => VirtualKeyCode::Question, + x11::keysym::XK_at => VirtualKeyCode::At, + x11::keysym::XK_A => VirtualKeyCode::A, + x11::keysym::XK_B => VirtualKeyCode::B, + x11::keysym::XK_C => VirtualKeyCode::C, + x11::keysym::XK_D => VirtualKeyCode::D, + x11::keysym::XK_E => VirtualKeyCode::E, + x11::keysym::XK_F => VirtualKeyCode::F, + x11::keysym::XK_G => VirtualKeyCode::G, + x11::keysym::XK_H => VirtualKeyCode::H, + x11::keysym::XK_I => VirtualKeyCode::I, + x11::keysym::XK_J => VirtualKeyCode::J, + x11::keysym::XK_K => VirtualKeyCode::K, + x11::keysym::XK_L => VirtualKeyCode::L, + x11::keysym::XK_M => VirtualKeyCode::M, + x11::keysym::XK_N => VirtualKeyCode::N, + x11::keysym::XK_O => VirtualKeyCode::O, + x11::keysym::XK_P => VirtualKeyCode::P, + x11::keysym::XK_Q => VirtualKeyCode::Q, + x11::keysym::XK_R => VirtualKeyCode::R, + x11::keysym::XK_S => VirtualKeyCode::S, + x11::keysym::XK_T => VirtualKeyCode::T, + x11::keysym::XK_U => VirtualKeyCode::U, + x11::keysym::XK_V => VirtualKeyCode::V, + x11::keysym::XK_W => VirtualKeyCode::W, + x11::keysym::XK_X => VirtualKeyCode::X, + x11::keysym::XK_Y => VirtualKeyCode::Y, + x11::keysym::XK_Z => VirtualKeyCode::Z, + x11::keysym::XK_bracketleft => VirtualKeyCode::LBracket, + x11::keysym::XK_backslash => VirtualKeyCode::Backslash, + x11::keysym::XK_bracketright => VirtualKeyCode::RBracket, + //x11::keysym::XK_asciicircum => VirtualKeyCode::Asciicircum, + //x11::keysym::XK_underscore => VirtualKeyCode::Underscore, + x11::keysym::XK_grave => VirtualKeyCode::Grave, + //x11::keysym::XK_quoteleft => VirtualKeyCode::Quoteleft, + x11::keysym::XK_a => VirtualKeyCode::A, + x11::keysym::XK_b => VirtualKeyCode::B, + x11::keysym::XK_c => VirtualKeyCode::C, + x11::keysym::XK_d => VirtualKeyCode::D, + x11::keysym::XK_e => VirtualKeyCode::E, + x11::keysym::XK_f => VirtualKeyCode::F, + x11::keysym::XK_g => VirtualKeyCode::G, + x11::keysym::XK_h => VirtualKeyCode::H, + x11::keysym::XK_i => VirtualKeyCode::I, + x11::keysym::XK_j => VirtualKeyCode::J, + x11::keysym::XK_k => VirtualKeyCode::K, + x11::keysym::XK_l => VirtualKeyCode::L, + x11::keysym::XK_m => VirtualKeyCode::M, + x11::keysym::XK_n => VirtualKeyCode::N, + x11::keysym::XK_o => VirtualKeyCode::O, + x11::keysym::XK_p => VirtualKeyCode::P, + x11::keysym::XK_q => VirtualKeyCode::Q, + x11::keysym::XK_r => VirtualKeyCode::R, + x11::keysym::XK_s => VirtualKeyCode::S, + x11::keysym::XK_t => VirtualKeyCode::T, + x11::keysym::XK_u => VirtualKeyCode::U, + x11::keysym::XK_v => VirtualKeyCode::V, + x11::keysym::XK_w => VirtualKeyCode::W, + x11::keysym::XK_x => VirtualKeyCode::X, + x11::keysym::XK_y => VirtualKeyCode::Y, + x11::keysym::XK_z => VirtualKeyCode::Z, + //x11::keysym::XK_braceleft => VirtualKeyCode::Braceleft, + //x11::keysym::XK_bar => VirtualKeyCode::Bar, + //x11::keysym::XK_braceright => VirtualKeyCode::Braceright, + //x11::keysym::XK_asciitilde => VirtualKeyCode::Asciitilde, + //x11::keysym::XK_nobreakspace => VirtualKeyCode::Nobreakspace, + //x11::keysym::XK_exclamdown => VirtualKeyCode::Exclamdown, + //x11::keysym::XK_cent => VirtualKeyCode::Cent, + //x11::keysym::XK_sterling => VirtualKeyCode::Sterling, + //x11::keysym::XK_currency => VirtualKeyCode::Currency, + //x11::keysym::XK_yen => VirtualKeyCode::Yen, + //x11::keysym::XK_brokenbar => VirtualKeyCode::Brokenbar, + //x11::keysym::XK_section => VirtualKeyCode::Section, + //x11::keysym::XK_diaeresis => VirtualKeyCode::Diaeresis, + //x11::keysym::XK_copyright => VirtualKeyCode::Copyright, + //x11::keysym::XK_ordfeminine => VirtualKeyCode::Ordfeminine, + //x11::keysym::XK_guillemotleft => VirtualKeyCode::Guillemotleft, + //x11::keysym::XK_notsign => VirtualKeyCode::Notsign, + //x11::keysym::XK_hyphen => VirtualKeyCode::Hyphen, + //x11::keysym::XK_registered => VirtualKeyCode::Registered, + //x11::keysym::XK_macron => VirtualKeyCode::Macron, + //x11::keysym::XK_degree => VirtualKeyCode::Degree, + //x11::keysym::XK_plusminus => VirtualKeyCode::Plusminus, + //x11::keysym::XK_twosuperior => VirtualKeyCode::Twosuperior, + //x11::keysym::XK_threesuperior => VirtualKeyCode::Threesuperior, + //x11::keysym::XK_acute => VirtualKeyCode::Acute, + //x11::keysym::XK_mu => VirtualKeyCode::Mu, + //x11::keysym::XK_paragraph => VirtualKeyCode::Paragraph, + //x11::keysym::XK_periodcentered => VirtualKeyCode::Periodcentered, + //x11::keysym::XK_cedilla => VirtualKeyCode::Cedilla, + //x11::keysym::XK_onesuperior => VirtualKeyCode::Onesuperior, + //x11::keysym::XK_masculine => VirtualKeyCode::Masculine, + //x11::keysym::XK_guillemotright => VirtualKeyCode::Guillemotright, + //x11::keysym::XK_onequarter => VirtualKeyCode::Onequarter, + //x11::keysym::XK_onehalf => VirtualKeyCode::Onehalf, + //x11::keysym::XK_threequarters => VirtualKeyCode::Threequarters, + //x11::keysym::XK_questiondown => VirtualKeyCode::Questiondown, + //x11::keysym::XK_Agrave => VirtualKeyCode::Agrave, + //x11::keysym::XK_Aacute => VirtualKeyCode::Aacute, + //x11::keysym::XK_Acircumflex => VirtualKeyCode::Acircumflex, + //x11::keysym::XK_Atilde => VirtualKeyCode::Atilde, + //x11::keysym::XK_Adiaeresis => VirtualKeyCode::Adiaeresis, + //x11::keysym::XK_Aring => VirtualKeyCode::Aring, + //x11::keysym::XK_AE => VirtualKeyCode::Ae, + //x11::keysym::XK_Ccedilla => VirtualKeyCode::Ccedilla, + //x11::keysym::XK_Egrave => VirtualKeyCode::Egrave, + //x11::keysym::XK_Eacute => VirtualKeyCode::Eacute, + //x11::keysym::XK_Ecircumflex => VirtualKeyCode::Ecircumflex, + //x11::keysym::XK_Ediaeresis => VirtualKeyCode::Ediaeresis, + //x11::keysym::XK_Igrave => VirtualKeyCode::Igrave, + //x11::keysym::XK_Iacute => VirtualKeyCode::Iacute, + //x11::keysym::XK_Icircumflex => VirtualKeyCode::Icircumflex, + //x11::keysym::XK_Idiaeresis => VirtualKeyCode::Idiaeresis, + //x11::keysym::XK_ETH => VirtualKeyCode::Eth, + //x11::keysym::XK_Eth => VirtualKeyCode::Eth, + //x11::keysym::XK_Ntilde => VirtualKeyCode::Ntilde, + //x11::keysym::XK_Ograve => VirtualKeyCode::Ograve, + //x11::keysym::XK_Oacute => VirtualKeyCode::Oacute, + //x11::keysym::XK_Ocircumflex => VirtualKeyCode::Ocircumflex, + //x11::keysym::XK_Otilde => VirtualKeyCode::Otilde, + //x11::keysym::XK_Odiaeresis => VirtualKeyCode::Odiaeresis, + //x11::keysym::XK_multiply => VirtualKeyCode::Multiply, + //x11::keysym::XK_Ooblique => VirtualKeyCode::Ooblique, + //x11::keysym::XK_Ugrave => VirtualKeyCode::Ugrave, + //x11::keysym::XK_Uacute => VirtualKeyCode::Uacute, + //x11::keysym::XK_Ucircumflex => VirtualKeyCode::Ucircumflex, + //x11::keysym::XK_Udiaeresis => VirtualKeyCode::Udiaeresis, + //x11::keysym::XK_Yacute => VirtualKeyCode::Yacute, + //x11::keysym::XK_THORN => VirtualKeyCode::Thorn, + //x11::keysym::XK_Thorn => VirtualKeyCode::Thorn, + //x11::keysym::XK_ssharp => VirtualKeyCode::Ssharp, + //x11::keysym::XK_agrave => VirtualKeyCode::Agrave, + //x11::keysym::XK_aacute => VirtualKeyCode::Aacute, + //x11::keysym::XK_acircumflex => VirtualKeyCode::Acircumflex, + //x11::keysym::XK_atilde => VirtualKeyCode::Atilde, + //x11::keysym::XK_adiaeresis => VirtualKeyCode::Adiaeresis, + //x11::keysym::XK_aring => VirtualKeyCode::Aring, + //x11::keysym::XK_ae => VirtualKeyCode::Ae, + //x11::keysym::XK_ccedilla => VirtualKeyCode::Ccedilla, + //x11::keysym::XK_egrave => VirtualKeyCode::Egrave, + //x11::keysym::XK_eacute => VirtualKeyCode::Eacute, + //x11::keysym::XK_ecircumflex => VirtualKeyCode::Ecircumflex, + //x11::keysym::XK_ediaeresis => VirtualKeyCode::Ediaeresis, + //x11::keysym::XK_igrave => VirtualKeyCode::Igrave, + //x11::keysym::XK_iacute => VirtualKeyCode::Iacute, + //x11::keysym::XK_icircumflex => VirtualKeyCode::Icircumflex, + //x11::keysym::XK_idiaeresis => VirtualKeyCode::Idiaeresis, + //x11::keysym::XK_eth => VirtualKeyCode::Eth, + //x11::keysym::XK_ntilde => VirtualKeyCode::Ntilde, + //x11::keysym::XK_ograve => VirtualKeyCode::Ograve, + //x11::keysym::XK_oacute => VirtualKeyCode::Oacute, + //x11::keysym::XK_ocircumflex => VirtualKeyCode::Ocircumflex, + //x11::keysym::XK_otilde => VirtualKeyCode::Otilde, + //x11::keysym::XK_odiaeresis => VirtualKeyCode::Odiaeresis, + //x11::keysym::XK_division => VirtualKeyCode::Division, + //x11::keysym::XK_oslash => VirtualKeyCode::Oslash, + //x11::keysym::XK_ugrave => VirtualKeyCode::Ugrave, + //x11::keysym::XK_uacute => VirtualKeyCode::Uacute, + //x11::keysym::XK_ucircumflex => VirtualKeyCode::Ucircumflex, + //x11::keysym::XK_udiaeresis => VirtualKeyCode::Udiaeresis, + //x11::keysym::XK_yacute => VirtualKeyCode::Yacute, + //x11::keysym::XK_thorn => VirtualKeyCode::Thorn, + //x11::keysym::XK_ydiaeresis => VirtualKeyCode::Ydiaeresis, + //x11::keysym::XK_Aogonek => VirtualKeyCode::Aogonek, + //x11::keysym::XK_breve => VirtualKeyCode::Breve, + //x11::keysym::XK_Lstroke => VirtualKeyCode::Lstroke, + //x11::keysym::XK_Lcaron => VirtualKeyCode::Lcaron, + //x11::keysym::XK_Sacute => VirtualKeyCode::Sacute, + //x11::keysym::XK_Scaron => VirtualKeyCode::Scaron, + //x11::keysym::XK_Scedilla => VirtualKeyCode::Scedilla, + //x11::keysym::XK_Tcaron => VirtualKeyCode::Tcaron, + //x11::keysym::XK_Zacute => VirtualKeyCode::Zacute, + //x11::keysym::XK_Zcaron => VirtualKeyCode::Zcaron, + //x11::keysym::XK_Zabovedot => VirtualKeyCode::Zabovedot, + //x11::keysym::XK_aogonek => VirtualKeyCode::Aogonek, + //x11::keysym::XK_ogonek => VirtualKeyCode::Ogonek, + //x11::keysym::XK_lstroke => VirtualKeyCode::Lstroke, + //x11::keysym::XK_lcaron => VirtualKeyCode::Lcaron, + //x11::keysym::XK_sacute => VirtualKeyCode::Sacute, + //x11::keysym::XK_caron => VirtualKeyCode::Caron, + //x11::keysym::XK_scaron => VirtualKeyCode::Scaron, + //x11::keysym::XK_scedilla => VirtualKeyCode::Scedilla, + //x11::keysym::XK_tcaron => VirtualKeyCode::Tcaron, + //x11::keysym::XK_zacute => VirtualKeyCode::Zacute, + //x11::keysym::XK_doubleacute => VirtualKeyCode::Doubleacute, + //x11::keysym::XK_zcaron => VirtualKeyCode::Zcaron, + //x11::keysym::XK_zabovedot => VirtualKeyCode::Zabovedot, + //x11::keysym::XK_Racute => VirtualKeyCode::Racute, + //x11::keysym::XK_Abreve => VirtualKeyCode::Abreve, + //x11::keysym::XK_Lacute => VirtualKeyCode::Lacute, + //x11::keysym::XK_Cacute => VirtualKeyCode::Cacute, + //x11::keysym::XK_Ccaron => VirtualKeyCode::Ccaron, + //x11::keysym::XK_Eogonek => VirtualKeyCode::Eogonek, + //x11::keysym::XK_Ecaron => VirtualKeyCode::Ecaron, + //x11::keysym::XK_Dcaron => VirtualKeyCode::Dcaron, + //x11::keysym::XK_Dstroke => VirtualKeyCode::Dstroke, + //x11::keysym::XK_Nacute => VirtualKeyCode::Nacute, + //x11::keysym::XK_Ncaron => VirtualKeyCode::Ncaron, + //x11::keysym::XK_Odoubleacute => VirtualKeyCode::Odoubleacute, + //x11::keysym::XK_Rcaron => VirtualKeyCode::Rcaron, + //x11::keysym::XK_Uring => VirtualKeyCode::Uring, + //x11::keysym::XK_Udoubleacute => VirtualKeyCode::Udoubleacute, + //x11::keysym::XK_Tcedilla => VirtualKeyCode::Tcedilla, + //x11::keysym::XK_racute => VirtualKeyCode::Racute, + //x11::keysym::XK_abreve => VirtualKeyCode::Abreve, + //x11::keysym::XK_lacute => VirtualKeyCode::Lacute, + //x11::keysym::XK_cacute => VirtualKeyCode::Cacute, + //x11::keysym::XK_ccaron => VirtualKeyCode::Ccaron, + //x11::keysym::XK_eogonek => VirtualKeyCode::Eogonek, + //x11::keysym::XK_ecaron => VirtualKeyCode::Ecaron, + //x11::keysym::XK_dcaron => VirtualKeyCode::Dcaron, + //x11::keysym::XK_dstroke => VirtualKeyCode::Dstroke, + //x11::keysym::XK_nacute => VirtualKeyCode::Nacute, + //x11::keysym::XK_ncaron => VirtualKeyCode::Ncaron, + //x11::keysym::XK_odoubleacute => VirtualKeyCode::Odoubleacute, + //x11::keysym::XK_udoubleacute => VirtualKeyCode::Udoubleacute, + //x11::keysym::XK_rcaron => VirtualKeyCode::Rcaron, + //x11::keysym::XK_uring => VirtualKeyCode::Uring, + //x11::keysym::XK_tcedilla => VirtualKeyCode::Tcedilla, + //x11::keysym::XK_abovedot => VirtualKeyCode::Abovedot, + //x11::keysym::XK_Hstroke => VirtualKeyCode::Hstroke, + //x11::keysym::XK_Hcircumflex => VirtualKeyCode::Hcircumflex, + //x11::keysym::XK_Iabovedot => VirtualKeyCode::Iabovedot, + //x11::keysym::XK_Gbreve => VirtualKeyCode::Gbreve, + //x11::keysym::XK_Jcircumflex => VirtualKeyCode::Jcircumflex, + //x11::keysym::XK_hstroke => VirtualKeyCode::Hstroke, + //x11::keysym::XK_hcircumflex => VirtualKeyCode::Hcircumflex, + //x11::keysym::XK_idotless => VirtualKeyCode::Idotless, + //x11::keysym::XK_gbreve => VirtualKeyCode::Gbreve, + //x11::keysym::XK_jcircumflex => VirtualKeyCode::Jcircumflex, + //x11::keysym::XK_Cabovedot => VirtualKeyCode::Cabovedot, + //x11::keysym::XK_Ccircumflex => VirtualKeyCode::Ccircumflex, + //x11::keysym::XK_Gabovedot => VirtualKeyCode::Gabovedot, + //x11::keysym::XK_Gcircumflex => VirtualKeyCode::Gcircumflex, + //x11::keysym::XK_Ubreve => VirtualKeyCode::Ubreve, + //x11::keysym::XK_Scircumflex => VirtualKeyCode::Scircumflex, + //x11::keysym::XK_cabovedot => VirtualKeyCode::Cabovedot, + //x11::keysym::XK_ccircumflex => VirtualKeyCode::Ccircumflex, + //x11::keysym::XK_gabovedot => VirtualKeyCode::Gabovedot, + //x11::keysym::XK_gcircumflex => VirtualKeyCode::Gcircumflex, + //x11::keysym::XK_ubreve => VirtualKeyCode::Ubreve, + //x11::keysym::XK_scircumflex => VirtualKeyCode::Scircumflex, + //x11::keysym::XK_kra => VirtualKeyCode::Kra, + //x11::keysym::XK_kappa => VirtualKeyCode::Kappa, + //x11::keysym::XK_Rcedilla => VirtualKeyCode::Rcedilla, + //x11::keysym::XK_Itilde => VirtualKeyCode::Itilde, + //x11::keysym::XK_Lcedilla => VirtualKeyCode::Lcedilla, + //x11::keysym::XK_Emacron => VirtualKeyCode::Emacron, + //x11::keysym::XK_Gcedilla => VirtualKeyCode::Gcedilla, + //x11::keysym::XK_Tslash => VirtualKeyCode::Tslash, + //x11::keysym::XK_rcedilla => VirtualKeyCode::Rcedilla, + //x11::keysym::XK_itilde => VirtualKeyCode::Itilde, + //x11::keysym::XK_lcedilla => VirtualKeyCode::Lcedilla, + //x11::keysym::XK_emacron => VirtualKeyCode::Emacron, + //x11::keysym::XK_gcedilla => VirtualKeyCode::Gcedilla, + //x11::keysym::XK_tslash => VirtualKeyCode::Tslash, + //x11::keysym::XK_ENG => VirtualKeyCode::Eng, + //x11::keysym::XK_eng => VirtualKeyCode::Eng, + //x11::keysym::XK_Amacron => VirtualKeyCode::Amacron, + //x11::keysym::XK_Iogonek => VirtualKeyCode::Iogonek, + //x11::keysym::XK_Eabovedot => VirtualKeyCode::Eabovedot, + //x11::keysym::XK_Imacron => VirtualKeyCode::Imacron, + //x11::keysym::XK_Ncedilla => VirtualKeyCode::Ncedilla, + //x11::keysym::XK_Omacron => VirtualKeyCode::Omacron, + //x11::keysym::XK_Kcedilla => VirtualKeyCode::Kcedilla, + //x11::keysym::XK_Uogonek => VirtualKeyCode::Uogonek, + //x11::keysym::XK_Utilde => VirtualKeyCode::Utilde, + //x11::keysym::XK_Umacron => VirtualKeyCode::Umacron, + //x11::keysym::XK_amacron => VirtualKeyCode::Amacron, + //x11::keysym::XK_iogonek => VirtualKeyCode::Iogonek, + //x11::keysym::XK_eabovedot => VirtualKeyCode::Eabovedot, + //x11::keysym::XK_imacron => VirtualKeyCode::Imacron, + //x11::keysym::XK_ncedilla => VirtualKeyCode::Ncedilla, + //x11::keysym::XK_omacron => VirtualKeyCode::Omacron, + //x11::keysym::XK_kcedilla => VirtualKeyCode::Kcedilla, + //x11::keysym::XK_uogonek => VirtualKeyCode::Uogonek, + //x11::keysym::XK_utilde => VirtualKeyCode::Utilde, + //x11::keysym::XK_umacron => VirtualKeyCode::Umacron, + //x11::keysym::XK_overline => VirtualKeyCode::Overline, + //x11::keysym::XK_kana_fullstop => VirtualKeyCode::Kana_fullstop, + //x11::keysym::XK_kana_openingbracket => VirtualKeyCode::Kana_openingbracket, + //x11::keysym::XK_kana_closingbracket => VirtualKeyCode::Kana_closingbracket, + //x11::keysym::XK_kana_comma => VirtualKeyCode::Kana_comma, + //x11::keysym::XK_kana_conjunctive => VirtualKeyCode::Kana_conjunctive, + //x11::keysym::XK_kana_middledot => VirtualKeyCode::Kana_middledot, + //x11::keysym::XK_kana_WO => VirtualKeyCode::Kana_wo, + //x11::keysym::XK_kana_a => VirtualKeyCode::Kana_a, + //x11::keysym::XK_kana_i => VirtualKeyCode::Kana_i, + //x11::keysym::XK_kana_u => VirtualKeyCode::Kana_u, + //x11::keysym::XK_kana_e => VirtualKeyCode::Kana_e, + //x11::keysym::XK_kana_o => VirtualKeyCode::Kana_o, + //x11::keysym::XK_kana_ya => VirtualKeyCode::Kana_ya, + //x11::keysym::XK_kana_yu => VirtualKeyCode::Kana_yu, + //x11::keysym::XK_kana_yo => VirtualKeyCode::Kana_yo, + //x11::keysym::XK_kana_tsu => VirtualKeyCode::Kana_tsu, + //x11::keysym::XK_kana_tu => VirtualKeyCode::Kana_tu, + //x11::keysym::XK_prolongedsound => VirtualKeyCode::Prolongedsound, + //x11::keysym::XK_kana_A => VirtualKeyCode::Kana_a, + //x11::keysym::XK_kana_I => VirtualKeyCode::Kana_i, + //x11::keysym::XK_kana_U => VirtualKeyCode::Kana_u, + //x11::keysym::XK_kana_E => VirtualKeyCode::Kana_e, + //x11::keysym::XK_kana_O => VirtualKeyCode::Kana_o, + //x11::keysym::XK_kana_KA => VirtualKeyCode::Kana_ka, + //x11::keysym::XK_kana_KI => VirtualKeyCode::Kana_ki, + //x11::keysym::XK_kana_KU => VirtualKeyCode::Kana_ku, + //x11::keysym::XK_kana_KE => VirtualKeyCode::Kana_ke, + //x11::keysym::XK_kana_KO => VirtualKeyCode::Kana_ko, + //x11::keysym::XK_kana_SA => VirtualKeyCode::Kana_sa, + //x11::keysym::XK_kana_SHI => VirtualKeyCode::Kana_shi, + //x11::keysym::XK_kana_SU => VirtualKeyCode::Kana_su, + //x11::keysym::XK_kana_SE => VirtualKeyCode::Kana_se, + //x11::keysym::XK_kana_SO => VirtualKeyCode::Kana_so, + //x11::keysym::XK_kana_TA => VirtualKeyCode::Kana_ta, + //x11::keysym::XK_kana_CHI => VirtualKeyCode::Kana_chi, + //x11::keysym::XK_kana_TI => VirtualKeyCode::Kana_ti, + //x11::keysym::XK_kana_TSU => VirtualKeyCode::Kana_tsu, + //x11::keysym::XK_kana_TU => VirtualKeyCode::Kana_tu, + //x11::keysym::XK_kana_TE => VirtualKeyCode::Kana_te, + //x11::keysym::XK_kana_TO => VirtualKeyCode::Kana_to, + //x11::keysym::XK_kana_NA => VirtualKeyCode::Kana_na, + //x11::keysym::XK_kana_NI => VirtualKeyCode::Kana_ni, + //x11::keysym::XK_kana_NU => VirtualKeyCode::Kana_nu, + //x11::keysym::XK_kana_NE => VirtualKeyCode::Kana_ne, + //x11::keysym::XK_kana_NO => VirtualKeyCode::Kana_no, + //x11::keysym::XK_kana_HA => VirtualKeyCode::Kana_ha, + //x11::keysym::XK_kana_HI => VirtualKeyCode::Kana_hi, + //x11::keysym::XK_kana_FU => VirtualKeyCode::Kana_fu, + //x11::keysym::XK_kana_HU => VirtualKeyCode::Kana_hu, + //x11::keysym::XK_kana_HE => VirtualKeyCode::Kana_he, + //x11::keysym::XK_kana_HO => VirtualKeyCode::Kana_ho, + //x11::keysym::XK_kana_MA => VirtualKeyCode::Kana_ma, + //x11::keysym::XK_kana_MI => VirtualKeyCode::Kana_mi, + //x11::keysym::XK_kana_MU => VirtualKeyCode::Kana_mu, + //x11::keysym::XK_kana_ME => VirtualKeyCode::Kana_me, + //x11::keysym::XK_kana_MO => VirtualKeyCode::Kana_mo, + //x11::keysym::XK_kana_YA => VirtualKeyCode::Kana_ya, + //x11::keysym::XK_kana_YU => VirtualKeyCode::Kana_yu, + //x11::keysym::XK_kana_YO => VirtualKeyCode::Kana_yo, + //x11::keysym::XK_kana_RA => VirtualKeyCode::Kana_ra, + //x11::keysym::XK_kana_RI => VirtualKeyCode::Kana_ri, + //x11::keysym::XK_kana_RU => VirtualKeyCode::Kana_ru, + //x11::keysym::XK_kana_RE => VirtualKeyCode::Kana_re, + //x11::keysym::XK_kana_RO => VirtualKeyCode::Kana_ro, + //x11::keysym::XK_kana_WA => VirtualKeyCode::Kana_wa, + //x11::keysym::XK_kana_N => VirtualKeyCode::Kana_n, + //x11::keysym::XK_voicedsound => VirtualKeyCode::Voicedsound, + //x11::keysym::XK_semivoicedsound => VirtualKeyCode::Semivoicedsound, + //x11::keysym::XK_kana_switch => VirtualKeyCode::Kana_switch, + //x11::keysym::XK_Arabic_comma => VirtualKeyCode::Arabic_comma, + //x11::keysym::XK_Arabic_semicolon => VirtualKeyCode::Arabic_semicolon, + //x11::keysym::XK_Arabic_question_mark => VirtualKeyCode::Arabic_question_mark, + //x11::keysym::XK_Arabic_hamza => VirtualKeyCode::Arabic_hamza, + //x11::keysym::XK_Arabic_maddaonalef => VirtualKeyCode::Arabic_maddaonalef, + //x11::keysym::XK_Arabic_hamzaonalef => VirtualKeyCode::Arabic_hamzaonalef, + //x11::keysym::XK_Arabic_hamzaonwaw => VirtualKeyCode::Arabic_hamzaonwaw, + //x11::keysym::XK_Arabic_hamzaunderalef => VirtualKeyCode::Arabic_hamzaunderalef, + //x11::keysym::XK_Arabic_hamzaonyeh => VirtualKeyCode::Arabic_hamzaonyeh, + //x11::keysym::XK_Arabic_alef => VirtualKeyCode::Arabic_alef, + //x11::keysym::XK_Arabic_beh => VirtualKeyCode::Arabic_beh, + //x11::keysym::XK_Arabic_tehmarbuta => VirtualKeyCode::Arabic_tehmarbuta, + //x11::keysym::XK_Arabic_teh => VirtualKeyCode::Arabic_teh, + //x11::keysym::XK_Arabic_theh => VirtualKeyCode::Arabic_theh, + //x11::keysym::XK_Arabic_jeem => VirtualKeyCode::Arabic_jeem, + //x11::keysym::XK_Arabic_hah => VirtualKeyCode::Arabic_hah, + //x11::keysym::XK_Arabic_khah => VirtualKeyCode::Arabic_khah, + //x11::keysym::XK_Arabic_dal => VirtualKeyCode::Arabic_dal, + //x11::keysym::XK_Arabic_thal => VirtualKeyCode::Arabic_thal, + //x11::keysym::XK_Arabic_ra => VirtualKeyCode::Arabic_ra, + //x11::keysym::XK_Arabic_zain => VirtualKeyCode::Arabic_zain, + //x11::keysym::XK_Arabic_seen => VirtualKeyCode::Arabic_seen, + //x11::keysym::XK_Arabic_sheen => VirtualKeyCode::Arabic_sheen, + //x11::keysym::XK_Arabic_sad => VirtualKeyCode::Arabic_sad, + //x11::keysym::XK_Arabic_dad => VirtualKeyCode::Arabic_dad, + //x11::keysym::XK_Arabic_tah => VirtualKeyCode::Arabic_tah, + //x11::keysym::XK_Arabic_zah => VirtualKeyCode::Arabic_zah, + //x11::keysym::XK_Arabic_ain => VirtualKeyCode::Arabic_ain, + //x11::keysym::XK_Arabic_ghain => VirtualKeyCode::Arabic_ghain, + //x11::keysym::XK_Arabic_tatweel => VirtualKeyCode::Arabic_tatweel, + //x11::keysym::XK_Arabic_feh => VirtualKeyCode::Arabic_feh, + //x11::keysym::XK_Arabic_qaf => VirtualKeyCode::Arabic_qaf, + //x11::keysym::XK_Arabic_kaf => VirtualKeyCode::Arabic_kaf, + //x11::keysym::XK_Arabic_lam => VirtualKeyCode::Arabic_lam, + //x11::keysym::XK_Arabic_meem => VirtualKeyCode::Arabic_meem, + //x11::keysym::XK_Arabic_noon => VirtualKeyCode::Arabic_noon, + //x11::keysym::XK_Arabic_ha => VirtualKeyCode::Arabic_ha, + //x11::keysym::XK_Arabic_heh => VirtualKeyCode::Arabic_heh, + //x11::keysym::XK_Arabic_waw => VirtualKeyCode::Arabic_waw, + //x11::keysym::XK_Arabic_alefmaksura => VirtualKeyCode::Arabic_alefmaksura, + //x11::keysym::XK_Arabic_yeh => VirtualKeyCode::Arabic_yeh, + //x11::keysym::XK_Arabic_fathatan => VirtualKeyCode::Arabic_fathatan, + //x11::keysym::XK_Arabic_dammatan => VirtualKeyCode::Arabic_dammatan, + //x11::keysym::XK_Arabic_kasratan => VirtualKeyCode::Arabic_kasratan, + //x11::keysym::XK_Arabic_fatha => VirtualKeyCode::Arabic_fatha, + //x11::keysym::XK_Arabic_damma => VirtualKeyCode::Arabic_damma, + //x11::keysym::XK_Arabic_kasra => VirtualKeyCode::Arabic_kasra, + //x11::keysym::XK_Arabic_shadda => VirtualKeyCode::Arabic_shadda, + //x11::keysym::XK_Arabic_sukun => VirtualKeyCode::Arabic_sukun, + //x11::keysym::XK_Arabic_switch => VirtualKeyCode::Arabic_switch, + //x11::keysym::XK_Serbian_dje => VirtualKeyCode::Serbian_dje, + //x11::keysym::XK_Macedonia_gje => VirtualKeyCode::Macedonia_gje, + //x11::keysym::XK_Cyrillic_io => VirtualKeyCode::Cyrillic_io, + //x11::keysym::XK_Ukrainian_ie => VirtualKeyCode::Ukrainian_ie, + //x11::keysym::XK_Ukranian_je => VirtualKeyCode::Ukranian_je, + //x11::keysym::XK_Macedonia_dse => VirtualKeyCode::Macedonia_dse, + //x11::keysym::XK_Ukrainian_i => VirtualKeyCode::Ukrainian_i, + //x11::keysym::XK_Ukranian_i => VirtualKeyCode::Ukranian_i, + //x11::keysym::XK_Ukrainian_yi => VirtualKeyCode::Ukrainian_yi, + //x11::keysym::XK_Ukranian_yi => VirtualKeyCode::Ukranian_yi, + //x11::keysym::XK_Cyrillic_je => VirtualKeyCode::Cyrillic_je, + //x11::keysym::XK_Serbian_je => VirtualKeyCode::Serbian_je, + //x11::keysym::XK_Cyrillic_lje => VirtualKeyCode::Cyrillic_lje, + //x11::keysym::XK_Serbian_lje => VirtualKeyCode::Serbian_lje, + //x11::keysym::XK_Cyrillic_nje => VirtualKeyCode::Cyrillic_nje, + //x11::keysym::XK_Serbian_nje => VirtualKeyCode::Serbian_nje, + //x11::keysym::XK_Serbian_tshe => VirtualKeyCode::Serbian_tshe, + //x11::keysym::XK_Macedonia_kje => VirtualKeyCode::Macedonia_kje, + //x11::keysym::XK_Byelorussian_shortu => VirtualKeyCode::Byelorussian_shortu, + //x11::keysym::XK_Cyrillic_dzhe => VirtualKeyCode::Cyrillic_dzhe, + //x11::keysym::XK_Serbian_dze => VirtualKeyCode::Serbian_dze, + //x11::keysym::XK_numerosign => VirtualKeyCode::Numerosign, + //x11::keysym::XK_Serbian_DJE => VirtualKeyCode::Serbian_dje, + //x11::keysym::XK_Macedonia_GJE => VirtualKeyCode::Macedonia_gje, + //x11::keysym::XK_Cyrillic_IO => VirtualKeyCode::Cyrillic_io, + //x11::keysym::XK_Ukrainian_IE => VirtualKeyCode::Ukrainian_ie, + //x11::keysym::XK_Ukranian_JE => VirtualKeyCode::Ukranian_je, + //x11::keysym::XK_Macedonia_DSE => VirtualKeyCode::Macedonia_dse, + //x11::keysym::XK_Ukrainian_I => VirtualKeyCode::Ukrainian_i, + //x11::keysym::XK_Ukranian_I => VirtualKeyCode::Ukranian_i, + //x11::keysym::XK_Ukrainian_YI => VirtualKeyCode::Ukrainian_yi, + //x11::keysym::XK_Ukranian_YI => VirtualKeyCode::Ukranian_yi, + //x11::keysym::XK_Cyrillic_JE => VirtualKeyCode::Cyrillic_je, + //x11::keysym::XK_Serbian_JE => VirtualKeyCode::Serbian_je, + //x11::keysym::XK_Cyrillic_LJE => VirtualKeyCode::Cyrillic_lje, + //x11::keysym::XK_Serbian_LJE => VirtualKeyCode::Serbian_lje, + //x11::keysym::XK_Cyrillic_NJE => VirtualKeyCode::Cyrillic_nje, + //x11::keysym::XK_Serbian_NJE => VirtualKeyCode::Serbian_nje, + //x11::keysym::XK_Serbian_TSHE => VirtualKeyCode::Serbian_tshe, + //x11::keysym::XK_Macedonia_KJE => VirtualKeyCode::Macedonia_kje, + //x11::keysym::XK_Byelorussian_SHORTU => VirtualKeyCode::Byelorussian_shortu, + //x11::keysym::XK_Cyrillic_DZHE => VirtualKeyCode::Cyrillic_dzhe, + //x11::keysym::XK_Serbian_DZE => VirtualKeyCode::Serbian_dze, + //x11::keysym::XK_Cyrillic_yu => VirtualKeyCode::Cyrillic_yu, + //x11::keysym::XK_Cyrillic_a => VirtualKeyCode::Cyrillic_a, + //x11::keysym::XK_Cyrillic_be => VirtualKeyCode::Cyrillic_be, + //x11::keysym::XK_Cyrillic_tse => VirtualKeyCode::Cyrillic_tse, + //x11::keysym::XK_Cyrillic_de => VirtualKeyCode::Cyrillic_de, + //x11::keysym::XK_Cyrillic_ie => VirtualKeyCode::Cyrillic_ie, + //x11::keysym::XK_Cyrillic_ef => VirtualKeyCode::Cyrillic_ef, + //x11::keysym::XK_Cyrillic_ghe => VirtualKeyCode::Cyrillic_ghe, + //x11::keysym::XK_Cyrillic_ha => VirtualKeyCode::Cyrillic_ha, + //x11::keysym::XK_Cyrillic_i => VirtualKeyCode::Cyrillic_i, + //x11::keysym::XK_Cyrillic_shorti => VirtualKeyCode::Cyrillic_shorti, + //x11::keysym::XK_Cyrillic_ka => VirtualKeyCode::Cyrillic_ka, + //x11::keysym::XK_Cyrillic_el => VirtualKeyCode::Cyrillic_el, + //x11::keysym::XK_Cyrillic_em => VirtualKeyCode::Cyrillic_em, + //x11::keysym::XK_Cyrillic_en => VirtualKeyCode::Cyrillic_en, + //x11::keysym::XK_Cyrillic_o => VirtualKeyCode::Cyrillic_o, + //x11::keysym::XK_Cyrillic_pe => VirtualKeyCode::Cyrillic_pe, + //x11::keysym::XK_Cyrillic_ya => VirtualKeyCode::Cyrillic_ya, + //x11::keysym::XK_Cyrillic_er => VirtualKeyCode::Cyrillic_er, + //x11::keysym::XK_Cyrillic_es => VirtualKeyCode::Cyrillic_es, + //x11::keysym::XK_Cyrillic_te => VirtualKeyCode::Cyrillic_te, + //x11::keysym::XK_Cyrillic_u => VirtualKeyCode::Cyrillic_u, + //x11::keysym::XK_Cyrillic_zhe => VirtualKeyCode::Cyrillic_zhe, + //x11::keysym::XK_Cyrillic_ve => VirtualKeyCode::Cyrillic_ve, + //x11::keysym::XK_Cyrillic_softsign => VirtualKeyCode::Cyrillic_softsign, + //x11::keysym::XK_Cyrillic_yeru => VirtualKeyCode::Cyrillic_yeru, + //x11::keysym::XK_Cyrillic_ze => VirtualKeyCode::Cyrillic_ze, + //x11::keysym::XK_Cyrillic_sha => VirtualKeyCode::Cyrillic_sha, + //x11::keysym::XK_Cyrillic_e => VirtualKeyCode::Cyrillic_e, + //x11::keysym::XK_Cyrillic_shcha => VirtualKeyCode::Cyrillic_shcha, + //x11::keysym::XK_Cyrillic_che => VirtualKeyCode::Cyrillic_che, + //x11::keysym::XK_Cyrillic_hardsign => VirtualKeyCode::Cyrillic_hardsign, + //x11::keysym::XK_Cyrillic_YU => VirtualKeyCode::Cyrillic_yu, + //x11::keysym::XK_Cyrillic_A => VirtualKeyCode::Cyrillic_a, + //x11::keysym::XK_Cyrillic_BE => VirtualKeyCode::Cyrillic_be, + //x11::keysym::XK_Cyrillic_TSE => VirtualKeyCode::Cyrillic_tse, + //x11::keysym::XK_Cyrillic_DE => VirtualKeyCode::Cyrillic_de, + //x11::keysym::XK_Cyrillic_IE => VirtualKeyCode::Cyrillic_ie, + //x11::keysym::XK_Cyrillic_EF => VirtualKeyCode::Cyrillic_ef, + //x11::keysym::XK_Cyrillic_GHE => VirtualKeyCode::Cyrillic_ghe, + //x11::keysym::XK_Cyrillic_HA => VirtualKeyCode::Cyrillic_ha, + //x11::keysym::XK_Cyrillic_I => VirtualKeyCode::Cyrillic_i, + //x11::keysym::XK_Cyrillic_SHORTI => VirtualKeyCode::Cyrillic_shorti, + //x11::keysym::XK_Cyrillic_KA => VirtualKeyCode::Cyrillic_ka, + //x11::keysym::XK_Cyrillic_EL => VirtualKeyCode::Cyrillic_el, + //x11::keysym::XK_Cyrillic_EM => VirtualKeyCode::Cyrillic_em, + //x11::keysym::XK_Cyrillic_EN => VirtualKeyCode::Cyrillic_en, + //x11::keysym::XK_Cyrillic_O => VirtualKeyCode::Cyrillic_o, + //x11::keysym::XK_Cyrillic_PE => VirtualKeyCode::Cyrillic_pe, + //x11::keysym::XK_Cyrillic_YA => VirtualKeyCode::Cyrillic_ya, + //x11::keysym::XK_Cyrillic_ER => VirtualKeyCode::Cyrillic_er, + //x11::keysym::XK_Cyrillic_ES => VirtualKeyCode::Cyrillic_es, + //x11::keysym::XK_Cyrillic_TE => VirtualKeyCode::Cyrillic_te, + //x11::keysym::XK_Cyrillic_U => VirtualKeyCode::Cyrillic_u, + //x11::keysym::XK_Cyrillic_ZHE => VirtualKeyCode::Cyrillic_zhe, + //x11::keysym::XK_Cyrillic_VE => VirtualKeyCode::Cyrillic_ve, + //x11::keysym::XK_Cyrillic_SOFTSIGN => VirtualKeyCode::Cyrillic_softsign, + //x11::keysym::XK_Cyrillic_YERU => VirtualKeyCode::Cyrillic_yeru, + //x11::keysym::XK_Cyrillic_ZE => VirtualKeyCode::Cyrillic_ze, + //x11::keysym::XK_Cyrillic_SHA => VirtualKeyCode::Cyrillic_sha, + //x11::keysym::XK_Cyrillic_E => VirtualKeyCode::Cyrillic_e, + //x11::keysym::XK_Cyrillic_SHCHA => VirtualKeyCode::Cyrillic_shcha, + //x11::keysym::XK_Cyrillic_CHE => VirtualKeyCode::Cyrillic_che, + //x11::keysym::XK_Cyrillic_HARDSIGN => VirtualKeyCode::Cyrillic_hardsign, + //x11::keysym::XK_Greek_ALPHAaccent => VirtualKeyCode::Greek_alphaaccent, + //x11::keysym::XK_Greek_EPSILONaccent => VirtualKeyCode::Greek_epsilonaccent, + //x11::keysym::XK_Greek_ETAaccent => VirtualKeyCode::Greek_etaaccent, + //x11::keysym::XK_Greek_IOTAaccent => VirtualKeyCode::Greek_iotaaccent, + //x11::keysym::XK_Greek_IOTAdiaeresis => VirtualKeyCode::Greek_iotadiaeresis, + //x11::keysym::XK_Greek_OMICRONaccent => VirtualKeyCode::Greek_omicronaccent, + //x11::keysym::XK_Greek_UPSILONaccent => VirtualKeyCode::Greek_upsilonaccent, + //x11::keysym::XK_Greek_UPSILONdieresis => VirtualKeyCode::Greek_upsilondieresis, + //x11::keysym::XK_Greek_OMEGAaccent => VirtualKeyCode::Greek_omegaaccent, + //x11::keysym::XK_Greek_accentdieresis => VirtualKeyCode::Greek_accentdieresis, + //x11::keysym::XK_Greek_horizbar => VirtualKeyCode::Greek_horizbar, + //x11::keysym::XK_Greek_alphaaccent => VirtualKeyCode::Greek_alphaaccent, + //x11::keysym::XK_Greek_epsilonaccent => VirtualKeyCode::Greek_epsilonaccent, + //x11::keysym::XK_Greek_etaaccent => VirtualKeyCode::Greek_etaaccent, + //x11::keysym::XK_Greek_iotaaccent => VirtualKeyCode::Greek_iotaaccent, + //x11::keysym::XK_Greek_iotadieresis => VirtualKeyCode::Greek_iotadieresis, + //x11::keysym::XK_Greek_iotaaccentdieresis => VirtualKeyCode::Greek_iotaaccentdieresis, + //x11::keysym::XK_Greek_omicronaccent => VirtualKeyCode::Greek_omicronaccent, + //x11::keysym::XK_Greek_upsilonaccent => VirtualKeyCode::Greek_upsilonaccent, + //x11::keysym::XK_Greek_upsilondieresis => VirtualKeyCode::Greek_upsilondieresis, + //x11::keysym::XK_Greek_upsilonaccentdieresis => VirtualKeyCode::Greek_upsilonaccentdieresis, + //x11::keysym::XK_Greek_omegaaccent => VirtualKeyCode::Greek_omegaaccent, + //x11::keysym::XK_Greek_ALPHA => VirtualKeyCode::Greek_alpha, + //x11::keysym::XK_Greek_BETA => VirtualKeyCode::Greek_beta, + //x11::keysym::XK_Greek_GAMMA => VirtualKeyCode::Greek_gamma, + //x11::keysym::XK_Greek_DELTA => VirtualKeyCode::Greek_delta, + //x11::keysym::XK_Greek_EPSILON => VirtualKeyCode::Greek_epsilon, + //x11::keysym::XK_Greek_ZETA => VirtualKeyCode::Greek_zeta, + //x11::keysym::XK_Greek_ETA => VirtualKeyCode::Greek_eta, + //x11::keysym::XK_Greek_THETA => VirtualKeyCode::Greek_theta, + //x11::keysym::XK_Greek_IOTA => VirtualKeyCode::Greek_iota, + //x11::keysym::XK_Greek_KAPPA => VirtualKeyCode::Greek_kappa, + //x11::keysym::XK_Greek_LAMDA => VirtualKeyCode::Greek_lamda, + //x11::keysym::XK_Greek_LAMBDA => VirtualKeyCode::Greek_lambda, + //x11::keysym::XK_Greek_MU => VirtualKeyCode::Greek_mu, + //x11::keysym::XK_Greek_NU => VirtualKeyCode::Greek_nu, + //x11::keysym::XK_Greek_XI => VirtualKeyCode::Greek_xi, + //x11::keysym::XK_Greek_OMICRON => VirtualKeyCode::Greek_omicron, + //x11::keysym::XK_Greek_PI => VirtualKeyCode::Greek_pi, + //x11::keysym::XK_Greek_RHO => VirtualKeyCode::Greek_rho, + //x11::keysym::XK_Greek_SIGMA => VirtualKeyCode::Greek_sigma, + //x11::keysym::XK_Greek_TAU => VirtualKeyCode::Greek_tau, + //x11::keysym::XK_Greek_UPSILON => VirtualKeyCode::Greek_upsilon, + //x11::keysym::XK_Greek_PHI => VirtualKeyCode::Greek_phi, + //x11::keysym::XK_Greek_CHI => VirtualKeyCode::Greek_chi, + //x11::keysym::XK_Greek_PSI => VirtualKeyCode::Greek_psi, + //x11::keysym::XK_Greek_OMEGA => VirtualKeyCode::Greek_omega, + //x11::keysym::XK_Greek_alpha => VirtualKeyCode::Greek_alpha, + //x11::keysym::XK_Greek_beta => VirtualKeyCode::Greek_beta, + //x11::keysym::XK_Greek_gamma => VirtualKeyCode::Greek_gamma, + //x11::keysym::XK_Greek_delta => VirtualKeyCode::Greek_delta, + //x11::keysym::XK_Greek_epsilon => VirtualKeyCode::Greek_epsilon, + //x11::keysym::XK_Greek_zeta => VirtualKeyCode::Greek_zeta, + //x11::keysym::XK_Greek_eta => VirtualKeyCode::Greek_eta, + //x11::keysym::XK_Greek_theta => VirtualKeyCode::Greek_theta, + //x11::keysym::XK_Greek_iota => VirtualKeyCode::Greek_iota, + //x11::keysym::XK_Greek_kappa => VirtualKeyCode::Greek_kappa, + //x11::keysym::XK_Greek_lamda => VirtualKeyCode::Greek_lamda, + //x11::keysym::XK_Greek_lambda => VirtualKeyCode::Greek_lambda, + //x11::keysym::XK_Greek_mu => VirtualKeyCode::Greek_mu, + //x11::keysym::XK_Greek_nu => VirtualKeyCode::Greek_nu, + //x11::keysym::XK_Greek_xi => VirtualKeyCode::Greek_xi, + //x11::keysym::XK_Greek_omicron => VirtualKeyCode::Greek_omicron, + //x11::keysym::XK_Greek_pi => VirtualKeyCode::Greek_pi, + //x11::keysym::XK_Greek_rho => VirtualKeyCode::Greek_rho, + //x11::keysym::XK_Greek_sigma => VirtualKeyCode::Greek_sigma, + //x11::keysym::XK_Greek_finalsmallsigma => VirtualKeyCode::Greek_finalsmallsigma, + //x11::keysym::XK_Greek_tau => VirtualKeyCode::Greek_tau, + //x11::keysym::XK_Greek_upsilon => VirtualKeyCode::Greek_upsilon, + //x11::keysym::XK_Greek_phi => VirtualKeyCode::Greek_phi, + //x11::keysym::XK_Greek_chi => VirtualKeyCode::Greek_chi, + //x11::keysym::XK_Greek_psi => VirtualKeyCode::Greek_psi, + //x11::keysym::XK_Greek_omega => VirtualKeyCode::Greek_omega, + //x11::keysym::XK_Greek_switch => VirtualKeyCode::Greek_switch, + //x11::keysym::XK_leftradical => VirtualKeyCode::Leftradical, + //x11::keysym::XK_topleftradical => VirtualKeyCode::Topleftradical, + //x11::keysym::XK_horizconnector => VirtualKeyCode::Horizconnector, + //x11::keysym::XK_topintegral => VirtualKeyCode::Topintegral, + //x11::keysym::XK_botintegral => VirtualKeyCode::Botintegral, + //x11::keysym::XK_vertconnector => VirtualKeyCode::Vertconnector, + //x11::keysym::XK_topleftsqbracket => VirtualKeyCode::Topleftsqbracket, + //x11::keysym::XK_botleftsqbracket => VirtualKeyCode::Botleftsqbracket, + //x11::keysym::XK_toprightsqbracket => VirtualKeyCode::Toprightsqbracket, + //x11::keysym::XK_botrightsqbracket => VirtualKeyCode::Botrightsqbracket, + //x11::keysym::XK_topleftparens => VirtualKeyCode::Topleftparens, + //x11::keysym::XK_botleftparens => VirtualKeyCode::Botleftparens, + //x11::keysym::XK_toprightparens => VirtualKeyCode::Toprightparens, + //x11::keysym::XK_botrightparens => VirtualKeyCode::Botrightparens, + //x11::keysym::XK_leftmiddlecurlybrace => VirtualKeyCode::Leftmiddlecurlybrace, + //x11::keysym::XK_rightmiddlecurlybrace => VirtualKeyCode::Rightmiddlecurlybrace, + //x11::keysym::XK_topleftsummation => VirtualKeyCode::Topleftsummation, + //x11::keysym::XK_botleftsummation => VirtualKeyCode::Botleftsummation, + //x11::keysym::XK_topvertsummationconnector => VirtualKeyCode::Topvertsummationconnector, + //x11::keysym::XK_botvertsummationconnector => VirtualKeyCode::Botvertsummationconnector, + //x11::keysym::XK_toprightsummation => VirtualKeyCode::Toprightsummation, + //x11::keysym::XK_botrightsummation => VirtualKeyCode::Botrightsummation, + //x11::keysym::XK_rightmiddlesummation => VirtualKeyCode::Rightmiddlesummation, + //x11::keysym::XK_lessthanequal => VirtualKeyCode::Lessthanequal, + //x11::keysym::XK_notequal => VirtualKeyCode::Notequal, + //x11::keysym::XK_greaterthanequal => VirtualKeyCode::Greaterthanequal, + //x11::keysym::XK_integral => VirtualKeyCode::Integral, + //x11::keysym::XK_therefore => VirtualKeyCode::Therefore, + //x11::keysym::XK_variation => VirtualKeyCode::Variation, + //x11::keysym::XK_infinity => VirtualKeyCode::Infinity, + //x11::keysym::XK_nabla => VirtualKeyCode::Nabla, + //x11::keysym::XK_approximate => VirtualKeyCode::Approximate, + //x11::keysym::XK_similarequal => VirtualKeyCode::Similarequal, + //x11::keysym::XK_ifonlyif => VirtualKeyCode::Ifonlyif, + //x11::keysym::XK_implies => VirtualKeyCode::Implies, + //x11::keysym::XK_identical => VirtualKeyCode::Identical, + //x11::keysym::XK_radical => VirtualKeyCode::Radical, + //x11::keysym::XK_includedin => VirtualKeyCode::Includedin, + //x11::keysym::XK_includes => VirtualKeyCode::Includes, + //x11::keysym::XK_intersection => VirtualKeyCode::Intersection, + //x11::keysym::XK_union => VirtualKeyCode::Union, + //x11::keysym::XK_logicaland => VirtualKeyCode::Logicaland, + //x11::keysym::XK_logicalor => VirtualKeyCode::Logicalor, + //x11::keysym::XK_partialderivative => VirtualKeyCode::Partialderivative, + //x11::keysym::XK_function => VirtualKeyCode::Function, + //x11::keysym::XK_leftarrow => VirtualKeyCode::Leftarrow, + //x11::keysym::XK_uparrow => VirtualKeyCode::Uparrow, + //x11::keysym::XK_rightarrow => VirtualKeyCode::Rightarrow, + //x11::keysym::XK_downarrow => VirtualKeyCode::Downarrow, + //x11::keysym::XK_blank => VirtualKeyCode::Blank, + //x11::keysym::XK_soliddiamond => VirtualKeyCode::Soliddiamond, + //x11::keysym::XK_checkerboard => VirtualKeyCode::Checkerboard, + //x11::keysym::XK_ht => VirtualKeyCode::Ht, + //x11::keysym::XK_ff => VirtualKeyCode::Ff, + //x11::keysym::XK_cr => VirtualKeyCode::Cr, + //x11::keysym::XK_lf => VirtualKeyCode::Lf, + //x11::keysym::XK_nl => VirtualKeyCode::Nl, + //x11::keysym::XK_vt => VirtualKeyCode::Vt, + //x11::keysym::XK_lowrightcorner => VirtualKeyCode::Lowrightcorner, + //x11::keysym::XK_uprightcorner => VirtualKeyCode::Uprightcorner, + //x11::keysym::XK_upleftcorner => VirtualKeyCode::Upleftcorner, + //x11::keysym::XK_lowleftcorner => VirtualKeyCode::Lowleftcorner, + //x11::keysym::XK_crossinglines => VirtualKeyCode::Crossinglines, + //x11::keysym::XK_horizlinescan1 => VirtualKeyCode::Horizlinescan1, + //x11::keysym::XK_horizlinescan3 => VirtualKeyCode::Horizlinescan3, + //x11::keysym::XK_horizlinescan5 => VirtualKeyCode::Horizlinescan5, + //x11::keysym::XK_horizlinescan7 => VirtualKeyCode::Horizlinescan7, + //x11::keysym::XK_horizlinescan9 => VirtualKeyCode::Horizlinescan9, + //x11::keysym::XK_leftt => VirtualKeyCode::Leftt, + //x11::keysym::XK_rightt => VirtualKeyCode::Rightt, + //x11::keysym::XK_bott => VirtualKeyCode::Bott, + //x11::keysym::XK_topt => VirtualKeyCode::Topt, + //x11::keysym::XK_vertbar => VirtualKeyCode::Vertbar, + //x11::keysym::XK_emspace => VirtualKeyCode::Emspace, + //x11::keysym::XK_enspace => VirtualKeyCode::Enspace, + //x11::keysym::XK_em3space => VirtualKeyCode::Em3space, + //x11::keysym::XK_em4space => VirtualKeyCode::Em4space, + //x11::keysym::XK_digitspace => VirtualKeyCode::Digitspace, + //x11::keysym::XK_punctspace => VirtualKeyCode::Punctspace, + //x11::keysym::XK_thinspace => VirtualKeyCode::Thinspace, + //x11::keysym::XK_hairspace => VirtualKeyCode::Hairspace, + //x11::keysym::XK_emdash => VirtualKeyCode::Emdash, + //x11::keysym::XK_endash => VirtualKeyCode::Endash, + //x11::keysym::XK_signifblank => VirtualKeyCode::Signifblank, + //x11::keysym::XK_ellipsis => VirtualKeyCode::Ellipsis, + //x11::keysym::XK_doubbaselinedot => VirtualKeyCode::Doubbaselinedot, + //x11::keysym::XK_onethird => VirtualKeyCode::Onethird, + //x11::keysym::XK_twothirds => VirtualKeyCode::Twothirds, + //x11::keysym::XK_onefifth => VirtualKeyCode::Onefifth, + //x11::keysym::XK_twofifths => VirtualKeyCode::Twofifths, + //x11::keysym::XK_threefifths => VirtualKeyCode::Threefifths, + //x11::keysym::XK_fourfifths => VirtualKeyCode::Fourfifths, + //x11::keysym::XK_onesixth => VirtualKeyCode::Onesixth, + //x11::keysym::XK_fivesixths => VirtualKeyCode::Fivesixths, + //x11::keysym::XK_careof => VirtualKeyCode::Careof, + //x11::keysym::XK_figdash => VirtualKeyCode::Figdash, + //x11::keysym::XK_leftanglebracket => VirtualKeyCode::Leftanglebracket, + //x11::keysym::XK_decimalpoint => VirtualKeyCode::Decimalpoint, + //x11::keysym::XK_rightanglebracket => VirtualKeyCode::Rightanglebracket, + //x11::keysym::XK_marker => VirtualKeyCode::Marker, + //x11::keysym::XK_oneeighth => VirtualKeyCode::Oneeighth, + //x11::keysym::XK_threeeighths => VirtualKeyCode::Threeeighths, + //x11::keysym::XK_fiveeighths => VirtualKeyCode::Fiveeighths, + //x11::keysym::XK_seveneighths => VirtualKeyCode::Seveneighths, + //x11::keysym::XK_trademark => VirtualKeyCode::Trademark, + //x11::keysym::XK_signaturemark => VirtualKeyCode::Signaturemark, + //x11::keysym::XK_trademarkincircle => VirtualKeyCode::Trademarkincircle, + //x11::keysym::XK_leftopentriangle => VirtualKeyCode::Leftopentriangle, + //x11::keysym::XK_rightopentriangle => VirtualKeyCode::Rightopentriangle, + //x11::keysym::XK_emopencircle => VirtualKeyCode::Emopencircle, + //x11::keysym::XK_emopenrectangle => VirtualKeyCode::Emopenrectangle, + //x11::keysym::XK_leftsinglequotemark => VirtualKeyCode::Leftsinglequotemark, + //x11::keysym::XK_rightsinglequotemark => VirtualKeyCode::Rightsinglequotemark, + //x11::keysym::XK_leftdoublequotemark => VirtualKeyCode::Leftdoublequotemark, + //x11::keysym::XK_rightdoublequotemark => VirtualKeyCode::Rightdoublequotemark, + //x11::keysym::XK_prescription => VirtualKeyCode::Prescription, + //x11::keysym::XK_minutes => VirtualKeyCode::Minutes, + //x11::keysym::XK_seconds => VirtualKeyCode::Seconds, + //x11::keysym::XK_latincross => VirtualKeyCode::Latincross, + //x11::keysym::XK_hexagram => VirtualKeyCode::Hexagram, + //x11::keysym::XK_filledrectbullet => VirtualKeyCode::Filledrectbullet, + //x11::keysym::XK_filledlefttribullet => VirtualKeyCode::Filledlefttribullet, + //x11::keysym::XK_filledrighttribullet => VirtualKeyCode::Filledrighttribullet, + //x11::keysym::XK_emfilledcircle => VirtualKeyCode::Emfilledcircle, + //x11::keysym::XK_emfilledrect => VirtualKeyCode::Emfilledrect, + //x11::keysym::XK_enopencircbullet => VirtualKeyCode::Enopencircbullet, + //x11::keysym::XK_enopensquarebullet => VirtualKeyCode::Enopensquarebullet, + //x11::keysym::XK_openrectbullet => VirtualKeyCode::Openrectbullet, + //x11::keysym::XK_opentribulletup => VirtualKeyCode::Opentribulletup, + //x11::keysym::XK_opentribulletdown => VirtualKeyCode::Opentribulletdown, + //x11::keysym::XK_openstar => VirtualKeyCode::Openstar, + //x11::keysym::XK_enfilledcircbullet => VirtualKeyCode::Enfilledcircbullet, + //x11::keysym::XK_enfilledsqbullet => VirtualKeyCode::Enfilledsqbullet, + //x11::keysym::XK_filledtribulletup => VirtualKeyCode::Filledtribulletup, + //x11::keysym::XK_filledtribulletdown => VirtualKeyCode::Filledtribulletdown, + //x11::keysym::XK_leftpointer => VirtualKeyCode::Leftpointer, + //x11::keysym::XK_rightpointer => VirtualKeyCode::Rightpointer, + //x11::keysym::XK_club => VirtualKeyCode::Club, + //x11::keysym::XK_diamond => VirtualKeyCode::Diamond, + //x11::keysym::XK_heart => VirtualKeyCode::Heart, + //x11::keysym::XK_maltesecross => VirtualKeyCode::Maltesecross, + //x11::keysym::XK_dagger => VirtualKeyCode::Dagger, + //x11::keysym::XK_doubledagger => VirtualKeyCode::Doubledagger, + //x11::keysym::XK_checkmark => VirtualKeyCode::Checkmark, + //x11::keysym::XK_ballotcross => VirtualKeyCode::Ballotcross, + //x11::keysym::XK_musicalsharp => VirtualKeyCode::Musicalsharp, + //x11::keysym::XK_musicalflat => VirtualKeyCode::Musicalflat, + //x11::keysym::XK_malesymbol => VirtualKeyCode::Malesymbol, + //x11::keysym::XK_femalesymbol => VirtualKeyCode::Femalesymbol, + //x11::keysym::XK_telephone => VirtualKeyCode::Telephone, + //x11::keysym::XK_telephonerecorder => VirtualKeyCode::Telephonerecorder, + //x11::keysym::XK_phonographcopyright => VirtualKeyCode::Phonographcopyright, + //x11::keysym::XK_caret => VirtualKeyCode::Caret, + //x11::keysym::XK_singlelowquotemark => VirtualKeyCode::Singlelowquotemark, + //x11::keysym::XK_doublelowquotemark => VirtualKeyCode::Doublelowquotemark, + //x11::keysym::XK_cursor => VirtualKeyCode::Cursor, + //x11::keysym::XK_leftcaret => VirtualKeyCode::Leftcaret, + //x11::keysym::XK_rightcaret => VirtualKeyCode::Rightcaret, + //x11::keysym::XK_downcaret => VirtualKeyCode::Downcaret, + //x11::keysym::XK_upcaret => VirtualKeyCode::Upcaret, + //x11::keysym::XK_overbar => VirtualKeyCode::Overbar, + //x11::keysym::XK_downtack => VirtualKeyCode::Downtack, + //x11::keysym::XK_upshoe => VirtualKeyCode::Upshoe, + //x11::keysym::XK_downstile => VirtualKeyCode::Downstile, + //x11::keysym::XK_underbar => VirtualKeyCode::Underbar, + //x11::keysym::XK_jot => VirtualKeyCode::Jot, + //x11::keysym::XK_quad => VirtualKeyCode::Quad, + //x11::keysym::XK_uptack => VirtualKeyCode::Uptack, + //x11::keysym::XK_circle => VirtualKeyCode::Circle, + //x11::keysym::XK_upstile => VirtualKeyCode::Upstile, + //x11::keysym::XK_downshoe => VirtualKeyCode::Downshoe, + //x11::keysym::XK_rightshoe => VirtualKeyCode::Rightshoe, + //x11::keysym::XK_leftshoe => VirtualKeyCode::Leftshoe, + //x11::keysym::XK_lefttack => VirtualKeyCode::Lefttack, + //x11::keysym::XK_righttack => VirtualKeyCode::Righttack, + //x11::keysym::XK_hebrew_doublelowline => VirtualKeyCode::Hebrew_doublelowline, + //x11::keysym::XK_hebrew_aleph => VirtualKeyCode::Hebrew_aleph, + //x11::keysym::XK_hebrew_bet => VirtualKeyCode::Hebrew_bet, + //x11::keysym::XK_hebrew_beth => VirtualKeyCode::Hebrew_beth, + //x11::keysym::XK_hebrew_gimel => VirtualKeyCode::Hebrew_gimel, + //x11::keysym::XK_hebrew_gimmel => VirtualKeyCode::Hebrew_gimmel, + //x11::keysym::XK_hebrew_dalet => VirtualKeyCode::Hebrew_dalet, + //x11::keysym::XK_hebrew_daleth => VirtualKeyCode::Hebrew_daleth, + //x11::keysym::XK_hebrew_he => VirtualKeyCode::Hebrew_he, + //x11::keysym::XK_hebrew_waw => VirtualKeyCode::Hebrew_waw, + //x11::keysym::XK_hebrew_zain => VirtualKeyCode::Hebrew_zain, + //x11::keysym::XK_hebrew_zayin => VirtualKeyCode::Hebrew_zayin, + //x11::keysym::XK_hebrew_chet => VirtualKeyCode::Hebrew_chet, + //x11::keysym::XK_hebrew_het => VirtualKeyCode::Hebrew_het, + //x11::keysym::XK_hebrew_tet => VirtualKeyCode::Hebrew_tet, + //x11::keysym::XK_hebrew_teth => VirtualKeyCode::Hebrew_teth, + //x11::keysym::XK_hebrew_yod => VirtualKeyCode::Hebrew_yod, + //x11::keysym::XK_hebrew_finalkaph => VirtualKeyCode::Hebrew_finalkaph, + //x11::keysym::XK_hebrew_kaph => VirtualKeyCode::Hebrew_kaph, + //x11::keysym::XK_hebrew_lamed => VirtualKeyCode::Hebrew_lamed, + //x11::keysym::XK_hebrew_finalmem => VirtualKeyCode::Hebrew_finalmem, + //x11::keysym::XK_hebrew_mem => VirtualKeyCode::Hebrew_mem, + //x11::keysym::XK_hebrew_finalnun => VirtualKeyCode::Hebrew_finalnun, + //x11::keysym::XK_hebrew_nun => VirtualKeyCode::Hebrew_nun, + //x11::keysym::XK_hebrew_samech => VirtualKeyCode::Hebrew_samech, + //x11::keysym::XK_hebrew_samekh => VirtualKeyCode::Hebrew_samekh, + //x11::keysym::XK_hebrew_ayin => VirtualKeyCode::Hebrew_ayin, + //x11::keysym::XK_hebrew_finalpe => VirtualKeyCode::Hebrew_finalpe, + //x11::keysym::XK_hebrew_pe => VirtualKeyCode::Hebrew_pe, + //x11::keysym::XK_hebrew_finalzade => VirtualKeyCode::Hebrew_finalzade, + //x11::keysym::XK_hebrew_finalzadi => VirtualKeyCode::Hebrew_finalzadi, + //x11::keysym::XK_hebrew_zade => VirtualKeyCode::Hebrew_zade, + //x11::keysym::XK_hebrew_zadi => VirtualKeyCode::Hebrew_zadi, + //x11::keysym::XK_hebrew_qoph => VirtualKeyCode::Hebrew_qoph, + //x11::keysym::XK_hebrew_kuf => VirtualKeyCode::Hebrew_kuf, + //x11::keysym::XK_hebrew_resh => VirtualKeyCode::Hebrew_resh, + //x11::keysym::XK_hebrew_shin => VirtualKeyCode::Hebrew_shin, + //x11::keysym::XK_hebrew_taw => VirtualKeyCode::Hebrew_taw, + //x11::keysym::XK_hebrew_taf => VirtualKeyCode::Hebrew_taf, + //x11::keysym::XK_Hebrew_switch => VirtualKeyCode::Hebrew_switch, + x11::keysym::XF86XK_Back => VirtualKeyCode::NavigateBackward, + x11::keysym::XF86XK_Forward => VirtualKeyCode::NavigateForward, + x11::keysym::XF86XK_Copy => VirtualKeyCode::Copy, + x11::keysym::XF86XK_Paste => VirtualKeyCode::Paste, + x11::keysym::XF86XK_Cut => VirtualKeyCode::Cut, + _ => return None, + }) +} + +pub fn virtual_keycode_to_keysym(keycode: VirtualKeyCode) -> Option { + Some(match keycode { + VirtualKeyCode::Back => x11::keysym::XK_BackSpace, + VirtualKeyCode::Tab => x11::keysym::XK_Tab, + //VirtualKeyCode::Linefeed => x11::keysym::XK_Linefeed, + //VirtualKeyCode::Clear => x11::keysym::XK_Clear, + VirtualKeyCode::Return => x11::keysym::XK_Return, + //VirtualKeyCode::Pause => x11::keysym::XK_Pause, + //VirtualKeyCode::Scroll_lock => x11::keysym::XK_Scroll_Lock, + //VirtualKeyCode::Sys_req => x11::keysym::XK_Sys_Req, + VirtualKeyCode::Escape => x11::keysym::XK_Escape, + VirtualKeyCode::Delete => x11::keysym::XK_Delete, + VirtualKeyCode::Compose => x11::keysym::XK_Multi_key, + //VirtualKeyCode::Kanji => x11::keysym::XK_Kanji, + //VirtualKeyCode::Muhenkan => x11::keysym::XK_Muhenkan, + //VirtualKeyCode::Henkan_mode => x11::keysym::XK_Henkan_Mode, + //VirtualKeyCode::Henkan => x11::keysym::XK_Henkan, + //VirtualKeyCode::Romaji => x11::keysym::XK_Romaji, + //VirtualKeyCode::Hiragana => x11::keysym::XK_Hiragana, + //VirtualKeyCode::Katakana => x11::keysym::XK_Katakana, + //VirtualKeyCode::Hiragana_katakana => x11::keysym::XK_Hiragana_Katakana, + //VirtualKeyCode::Zenkaku => x11::keysym::XK_Zenkaku, + //VirtualKeyCode::Hankaku => x11::keysym::XK_Hankaku, + //VirtualKeyCode::Zenkaku_hankaku => x11::keysym::XK_Zenkaku_Hankaku, + //VirtualKeyCode::Touroku => x11::keysym::XK_Touroku, + //VirtualKeyCode::Massyo => x11::keysym::XK_Massyo, + //VirtualKeyCode::Kana_lock => x11::keysym::XK_Kana_Lock, + //VirtualKeyCode::Kana_shift => x11::keysym::XK_Kana_Shift, + //VirtualKeyCode::Eisu_shift => x11::keysym::XK_Eisu_Shift, + //VirtualKeyCode::Eisu_toggle => x11::keysym::XK_Eisu_toggle, + VirtualKeyCode::Home => x11::keysym::XK_Home, + VirtualKeyCode::Left => x11::keysym::XK_Left, + VirtualKeyCode::Up => x11::keysym::XK_Up, + VirtualKeyCode::Right => x11::keysym::XK_Right, + VirtualKeyCode::Down => x11::keysym::XK_Down, + //VirtualKeyCode::Prior => x11::keysym::XK_Prior, + VirtualKeyCode::PageUp => x11::keysym::XK_Page_Up, + //VirtualKeyCode::Next => x11::keysym::XK_Next, + VirtualKeyCode::PageDown => x11::keysym::XK_Page_Down, + VirtualKeyCode::End => x11::keysym::XK_End, + //VirtualKeyCode::Begin => x11::keysym::XK_Begin, + VirtualKeyCode::LWin => x11::keysym::XK_Win_L, + VirtualKeyCode::RWin => x11::keysym::XK_Win_R, + //VirtualKeyCode::App => x11::keysym::XK_App, + //VirtualKeyCode::Select => x11::keysym::XK_Select, + VirtualKeyCode::Print => x11::keysym::XK_Print, + //VirtualKeyCode::Execute => x11::keysym::XK_Execute, + VirtualKeyCode::Insert => x11::keysym::XK_Insert, + //VirtualKeyCode::Undo => x11::keysym::XK_Undo, + //VirtualKeyCode::Redo => x11::keysym::XK_Redo, + //VirtualKeyCode::Menu => x11::keysym::XK_Menu, + //VirtualKeyCode::Find => x11::keysym::XK_Find, + //VirtualKeyCode::Cancel => x11::keysym::XK_Cancel, + //VirtualKeyCode::Help => x11::keysym::XK_Help, + //VirtualKeyCode::Break => x11::keysym::XK_Break, + //VirtualKeyCode::Mode_switch => x11::keysym::XK_Mode_switch, + //VirtualKeyCode::Script_switch => x11::keysym::XK_script_switch, + //VirtualKeyCode::Num_lock => x11::keysym::XK_Num_Lock, + //VirtualKeyCode::Kp_space => x11::keysym::XK_KP_Space, + //VirtualKeyCode::Kp_tab => x11::keysym::XK_KP_Tab, + //VirtualKeyCode::Kp_enter => x11::keysym::XK_KP_Enter, + //VirtualKeyCode::Kp_f1 => x11::keysym::XK_KP_F1, + //VirtualKeyCode::Kp_f2 => x11::keysym::XK_KP_F2, + //VirtualKeyCode::Kp_f3 => x11::keysym::XK_KP_F3, + //VirtualKeyCode::Kp_f4 => x11::keysym::XK_KP_F4, + VirtualKeyCode::Home => x11::keysym::XK_KP_Home, + VirtualKeyCode::Left => x11::keysym::XK_KP_Left, + VirtualKeyCode::Up => x11::keysym::XK_KP_Up, + VirtualKeyCode::Right => x11::keysym::XK_KP_Right, + VirtualKeyCode::Down => x11::keysym::XK_KP_Down, + //VirtualKeyCode::Kp_prior => x11::keysym::XK_KP_Prior, + VirtualKeyCode::PageUp => x11::keysym::XK_KP_Page_Up, + //VirtualKeyCode::Kp_next => x11::keysym::XK_KP_Next, + VirtualKeyCode::PageDown => x11::keysym::XK_KP_Page_Down, + VirtualKeyCode::End => x11::keysym::XK_KP_End, + //VirtualKeyCode::Kp_begin => x11::keysym::XK_KP_Begin, + VirtualKeyCode::Insert => x11::keysym::XK_KP_Insert, + VirtualKeyCode::Delete => x11::keysym::XK_KP_Delete, + VirtualKeyCode::NumpadEquals => x11::keysym::XK_KP_Equal, + VirtualKeyCode::NumpadMultiply => x11::keysym::XK_KP_Multiply, + VirtualKeyCode::NumpadAdd => x11::keysym::XK_KP_Add, + //VirtualKeyCode::Kp_separator => x11::keysym::XK_KP_Separator, + VirtualKeyCode::NumpadSubtract => x11::keysym::XK_KP_Subtract, + VirtualKeyCode::NumpadDecimal => x11::keysym::XK_KP_Decimal, + VirtualKeyCode::NumpadDivide => x11::keysym::XK_KP_Divide, + VirtualKeyCode::Numpad0 => x11::keysym::XK_KP_0, + VirtualKeyCode::Numpad1 => x11::keysym::XK_KP_1, + VirtualKeyCode::Numpad2 => x11::keysym::XK_KP_2, + VirtualKeyCode::Numpad3 => x11::keysym::XK_KP_3, + VirtualKeyCode::Numpad4 => x11::keysym::XK_KP_4, + VirtualKeyCode::Numpad5 => x11::keysym::XK_KP_5, + VirtualKeyCode::Numpad6 => x11::keysym::XK_KP_6, + VirtualKeyCode::Numpad7 => x11::keysym::XK_KP_7, + VirtualKeyCode::Numpad8 => x11::keysym::XK_KP_8, + VirtualKeyCode::Numpad9 => x11::keysym::XK_KP_9, + VirtualKeyCode::F1 => x11::keysym::XK_F1, + VirtualKeyCode::F2 => x11::keysym::XK_F2, + VirtualKeyCode::F3 => x11::keysym::XK_F3, + VirtualKeyCode::F4 => x11::keysym::XK_F4, + VirtualKeyCode::F5 => x11::keysym::XK_F5, + VirtualKeyCode::F6 => x11::keysym::XK_F6, + VirtualKeyCode::F7 => x11::keysym::XK_F7, + VirtualKeyCode::F8 => x11::keysym::XK_F8, + VirtualKeyCode::F9 => x11::keysym::XK_F9, + VirtualKeyCode::F10 => x11::keysym::XK_F10, + VirtualKeyCode::F11 => x11::keysym::XK_F11, + //VirtualKeyCode::L1 => x11::keysym::XK_L1, + VirtualKeyCode::F12 => x11::keysym::XK_F12, + //VirtualKeyCode::L2 => x11::keysym::XK_L2, + VirtualKeyCode::F13 => x11::keysym::XK_F13, + //VirtualKeyCode::L3 => x11::keysym::XK_L3, + VirtualKeyCode::F14 => x11::keysym::XK_F14, + //VirtualKeyCode::L4 => x11::keysym::XK_L4, + VirtualKeyCode::F15 => x11::keysym::XK_F15, + //VirtualKeyCode::L5 => x11::keysym::XK_L5, + VirtualKeyCode::F16 => x11::keysym::XK_F16, + //VirtualKeyCode::L6 => x11::keysym::XK_L6, + VirtualKeyCode::F17 => x11::keysym::XK_F17, + //VirtualKeyCode::L7 => x11::keysym::XK_L7, + VirtualKeyCode::F18 => x11::keysym::XK_F18, + //VirtualKeyCode::L8 => x11::keysym::XK_L8, + VirtualKeyCode::F19 => x11::keysym::XK_F19, + //VirtualKeyCode::L9 => x11::keysym::XK_L9, + VirtualKeyCode::F20 => x11::keysym::XK_F20, + //VirtualKeyCode::L10 => x11::keysym::XK_L10, + VirtualKeyCode::F21 => x11::keysym::XK_F21, + //VirtualKeyCode::R1 => x11::keysym::XK_R1, + VirtualKeyCode::F22 => x11::keysym::XK_F22, + //VirtualKeyCode::R2 => x11::keysym::XK_R2, + VirtualKeyCode::F23 => x11::keysym::XK_F23, + //VirtualKeyCode::R3 => x11::keysym::XK_R3, + VirtualKeyCode::F24 => x11::keysym::XK_F24, + //VirtualKeyCode::R4 => x11::keysym::XK_R4, + //VirtualKeyCode::F25 => x11::keysym::XK_F25, + //VirtualKeyCode::R5 => x11::keysym::XK_R5, + //VirtualKeyCode::F26 => x11::keysym::XK_F26, + //VirtualKeyCode::R6 => x11::keysym::XK_R6, + //VirtualKeyCode::F27 => x11::keysym::XK_F27, + //VirtualKeyCode::R7 => x11::keysym::XK_R7, + //VirtualKeyCode::F28 => x11::keysym::XK_F28, + //VirtualKeyCode::R8 => x11::keysym::XK_R8, + //VirtualKeyCode::F29 => x11::keysym::XK_F29, + //VirtualKeyCode::R9 => x11::keysym::XK_R9, + //VirtualKeyCode::F30 => x11::keysym::XK_F30, + //VirtualKeyCode::R10 => x11::keysym::XK_R10, + //VirtualKeyCode::F31 => x11::keysym::XK_F31, + //VirtualKeyCode::R11 => x11::keysym::XK_R11, + //VirtualKeyCode::F32 => x11::keysym::XK_F32, + //VirtualKeyCode::R12 => x11::keysym::XK_R12, + //VirtualKeyCode::F33 => x11::keysym::XK_F33, + //VirtualKeyCode::R13 => x11::keysym::XK_R13, + //VirtualKeyCode::F34 => x11::keysym::XK_F34, + //VirtualKeyCode::R14 => x11::keysym::XK_R14, + //VirtualKeyCode::F35 => x11::keysym::XK_F35, + //VirtualKeyCode::R15 => x11::keysym::XK_R15, + VirtualKeyCode::LShift => x11::keysym::XK_Shift_L, + VirtualKeyCode::RShift => x11::keysym::XK_Shift_R, + VirtualKeyCode::LControl => x11::keysym::XK_Control_L, + VirtualKeyCode::RControl => x11::keysym::XK_Control_R, + //VirtualKeyCode::Caps_lock => x11::keysym::XK_Caps_Lock, + //VirtualKeyCode::Shift_lock => x11::keysym::XK_Shift_Lock, + //VirtualKeyCode::Meta_l => x11::keysym::XK_Meta_L, + //VirtualKeyCode::Meta_r => x11::keysym::XK_Meta_R, + VirtualKeyCode::LAlt => x11::keysym::XK_Alt_L, + VirtualKeyCode::RAlt => x11::keysym::XK_Alt_R, + //VirtualKeyCode::Super_l => x11::keysym::XK_Super_L, + //VirtualKeyCode::Super_r => x11::keysym::XK_Super_R, + //VirtualKeyCode::Hyper_l => x11::keysym::XK_Hyper_L, + //VirtualKeyCode::Hyper_r => x11::keysym::XK_Hyper_R, + VirtualKeyCode::Tab => x11::keysym::XK_ISO_Left_Tab, + VirtualKeyCode::Space => x11::keysym::XK_space, + //VirtualKeyCode::Exclam => x11::keysym::XK_exclam, + //VirtualKeyCode::Quotedbl => x11::keysym::XK_quotedbl, + //VirtualKeyCode::Numbersign => x11::keysym::XK_numbersign, + //VirtualKeyCode::Dollar => x11::keysym::XK_dollar, + //VirtualKeyCode::Percent => x11::keysym::XK_percent, + //VirtualKeyCode::Ampersand => x11::keysym::XK_ampersand, + VirtualKeyCode::Apostrophe => x11::keysym::XK_apostrophe, + //VirtualKeyCode::Quoteright => x11::keysym::XK_quoteright, + //VirtualKeyCode::Parenleft => x11::keysym::XK_parenleft, + //VirtualKeyCode::Parenright => x11::keysym::XK_parenright, + VirtualKeyCode::Asterisk => x11::keysym::XK_asterisk, + VirtualKeyCode::Plus => x11::keysym::XK_plus, + VirtualKeyCode::Comma => x11::keysym::XK_comma, + VirtualKeyCode::Minus => x11::keysym::XK_minus, + VirtualKeyCode::Period => x11::keysym::XK_period, + VirtualKeyCode::Slash => x11::keysym::XK_slash, + VirtualKeyCode::Zero => x11::keysym::XK_0, + VirtualKeyCode::One => x11::keysym::XK_1, + VirtualKeyCode::Two => x11::keysym::XK_2, + VirtualKeyCode::Three => x11::keysym::XK_3, + VirtualKeyCode::Four => x11::keysym::XK_4, + VirtualKeyCode::Five => x11::keysym::XK_5, + VirtualKeyCode::Six => x11::keysym::XK_6, + VirtualKeyCode::Seven => x11::keysym::XK_7, + VirtualKeyCode::Eight => x11::keysym::XK_8, + VirtualKeyCode::Nine => x11::keysym::XK_9, + VirtualKeyCode::Colon => x11::keysym::XK_colon, + VirtualKeyCode::Semicolon => x11::keysym::XK_semicolon, + //VirtualKeyCode::Less => x11::keysym::XK_less, + VirtualKeyCode::Equals => x11::keysym::XK_equal, + //VirtualKeyCode::Greater => x11::keysym::XK_greater, + //VirtualKeyCode::Question => x11::keysym::XK_question, + VirtualKeyCode::At => x11::keysym::XK_at, + VirtualKeyCode::A => x11::keysym::XK_A, + VirtualKeyCode::B => x11::keysym::XK_B, + VirtualKeyCode::C => x11::keysym::XK_C, + VirtualKeyCode::D => x11::keysym::XK_D, + VirtualKeyCode::E => x11::keysym::XK_E, + VirtualKeyCode::F => x11::keysym::XK_F, + VirtualKeyCode::G => x11::keysym::XK_G, + VirtualKeyCode::H => x11::keysym::XK_H, + VirtualKeyCode::I => x11::keysym::XK_I, + VirtualKeyCode::J => x11::keysym::XK_J, + VirtualKeyCode::K => x11::keysym::XK_K, + VirtualKeyCode::L => x11::keysym::XK_L, + VirtualKeyCode::M => x11::keysym::XK_M, + VirtualKeyCode::N => x11::keysym::XK_N, + VirtualKeyCode::O => x11::keysym::XK_O, + VirtualKeyCode::P => x11::keysym::XK_P, + VirtualKeyCode::Q => x11::keysym::XK_Q, + VirtualKeyCode::R => x11::keysym::XK_R, + VirtualKeyCode::S => x11::keysym::XK_S, + VirtualKeyCode::T => x11::keysym::XK_T, + VirtualKeyCode::U => x11::keysym::XK_U, + VirtualKeyCode::V => x11::keysym::XK_V, + VirtualKeyCode::W => x11::keysym::XK_W, + VirtualKeyCode::X => x11::keysym::XK_X, + VirtualKeyCode::Y => x11::keysym::XK_Y, + VirtualKeyCode::Z => x11::keysym::XK_Z, + VirtualKeyCode::LBracket => x11::keysym::XK_bracketleft, + VirtualKeyCode::Backslash => x11::keysym::XK_backslash, + VirtualKeyCode::RBracket => x11::keysym::XK_bracketright, + //VirtualKeyCode::Asciicircum => x11::keysym::XK_asciicircum, + //VirtualKeyCode::Underscore => x11::keysym::XK_underscore, + VirtualKeyCode::Grave => x11::keysym::XK_grave, + //VirtualKeyCode::Quoteleft => x11::keysym::XK_quoteleft, + VirtualKeyCode::A => x11::keysym::XK_a, + VirtualKeyCode::B => x11::keysym::XK_b, + VirtualKeyCode::C => x11::keysym::XK_c, + VirtualKeyCode::D => x11::keysym::XK_d, + VirtualKeyCode::E => x11::keysym::XK_e, + VirtualKeyCode::F => x11::keysym::XK_f, + VirtualKeyCode::G => x11::keysym::XK_g, + VirtualKeyCode::H => x11::keysym::XK_h, + VirtualKeyCode::I => x11::keysym::XK_i, + VirtualKeyCode::J => x11::keysym::XK_j, + VirtualKeyCode::K => x11::keysym::XK_k, + VirtualKeyCode::L => x11::keysym::XK_l, + VirtualKeyCode::M => x11::keysym::XK_m, + VirtualKeyCode::N => x11::keysym::XK_n, + VirtualKeyCode::O => x11::keysym::XK_o, + VirtualKeyCode::P => x11::keysym::XK_p, + VirtualKeyCode::Q => x11::keysym::XK_q, + VirtualKeyCode::R => x11::keysym::XK_r, + VirtualKeyCode::S => x11::keysym::XK_s, + VirtualKeyCode::T => x11::keysym::XK_t, + VirtualKeyCode::U => x11::keysym::XK_u, + VirtualKeyCode::V => x11::keysym::XK_v, + VirtualKeyCode::W => x11::keysym::XK_w, + VirtualKeyCode::X => x11::keysym::XK_x, + VirtualKeyCode::Y => x11::keysym::XK_y, + VirtualKeyCode::Z => x11::keysym::XK_z, + //VirtualKeyCode::Braceleft => x11::keysym::XK_braceleft, + //VirtualKeyCode::Bar => x11::keysym::XK_bar, + //VirtualKeyCode::Braceright => x11::keysym::XK_braceright, + //VirtualKeyCode::Asciitilde => x11::keysym::XK_asciitilde, + //VirtualKeyCode::Nobreakspace => x11::keysym::XK_nobreakspace, + //VirtualKeyCode::Exclamdown => x11::keysym::XK_exclamdown, + //VirtualKeyCode::Cent => x11::keysym::XK_cent, + //VirtualKeyCode::Sterling => x11::keysym::XK_sterling, + //VirtualKeyCode::Currency => x11::keysym::XK_currency, + //VirtualKeyCode::Yen => x11::keysym::XK_yen, + //VirtualKeyCode::Brokenbar => x11::keysym::XK_brokenbar, + //VirtualKeyCode::Section => x11::keysym::XK_section, + //VirtualKeyCode::Diaeresis => x11::keysym::XK_diaeresis, + //VirtualKeyCode::Copyright => x11::keysym::XK_copyright, + //VirtualKeyCode::Ordfeminine => x11::keysym::XK_ordfeminine, + //VirtualKeyCode::Guillemotleft => x11::keysym::XK_guillemotleft, + //VirtualKeyCode::Notsign => x11::keysym::XK_notsign, + //VirtualKeyCode::Hyphen => x11::keysym::XK_hyphen, + //VirtualKeyCode::Registered => x11::keysym::XK_registered, + //VirtualKeyCode::Macron => x11::keysym::XK_macron, + //VirtualKeyCode::Degree => x11::keysym::XK_degree, + //VirtualKeyCode::Plusminus => x11::keysym::XK_plusminus, + //VirtualKeyCode::Twosuperior => x11::keysym::XK_twosuperior, + //VirtualKeyCode::Threesuperior => x11::keysym::XK_threesuperior, + //VirtualKeyCode::Acute => x11::keysym::XK_acute, + //VirtualKeyCode::Mu => x11::keysym::XK_mu, + //VirtualKeyCode::Paragraph => x11::keysym::XK_paragraph, + //VirtualKeyCode::Periodcentered => x11::keysym::XK_periodcentered, + //VirtualKeyCode::Cedilla => x11::keysym::XK_cedilla, + //VirtualKeyCode::Onesuperior => x11::keysym::XK_onesuperior, + //VirtualKeyCode::Masculine => x11::keysym::XK_masculine, + //VirtualKeyCode::Guillemotright => x11::keysym::XK_guillemotright, + //VirtualKeyCode::Onequarter => x11::keysym::XK_onequarter, + //VirtualKeyCode::Onehalf => x11::keysym::XK_onehalf, + //VirtualKeyCode::Threequarters => x11::keysym::XK_threequarters, + //VirtualKeyCode::Questiondown => x11::keysym::XK_questiondown, + //VirtualKeyCode::Agrave => x11::keysym::XK_Agrave, + //VirtualKeyCode::Aacute => x11::keysym::XK_Aacute, + //VirtualKeyCode::Acircumflex => x11::keysym::XK_Acircumflex, + //VirtualKeyCode::Atilde => x11::keysym::XK_Atilde, + //VirtualKeyCode::Adiaeresis => x11::keysym::XK_Adiaeresis, + //VirtualKeyCode::Aring => x11::keysym::XK_Aring, + //VirtualKeyCode::Ae => x11::keysym::XK_AE, + //VirtualKeyCode::Ccedilla => x11::keysym::XK_Ccedilla, + //VirtualKeyCode::Egrave => x11::keysym::XK_Egrave, + //VirtualKeyCode::Eacute => x11::keysym::XK_Eacute, + //VirtualKeyCode::Ecircumflex => x11::keysym::XK_Ecircumflex, + //VirtualKeyCode::Ediaeresis => x11::keysym::XK_Ediaeresis, + //VirtualKeyCode::Igrave => x11::keysym::XK_Igrave, + //VirtualKeyCode::Iacute => x11::keysym::XK_Iacute, + //VirtualKeyCode::Icircumflex => x11::keysym::XK_Icircumflex, + //VirtualKeyCode::Idiaeresis => x11::keysym::XK_Idiaeresis, + //VirtualKeyCode::Eth => x11::keysym::XK_ETH, + //VirtualKeyCode::Eth => x11::keysym::XK_Eth, + //VirtualKeyCode::Ntilde => x11::keysym::XK_Ntilde, + //VirtualKeyCode::Ograve => x11::keysym::XK_Ograve, + //VirtualKeyCode::Oacute => x11::keysym::XK_Oacute, + //VirtualKeyCode::Ocircumflex => x11::keysym::XK_Ocircumflex, + //VirtualKeyCode::Otilde => x11::keysym::XK_Otilde, + //VirtualKeyCode::Odiaeresis => x11::keysym::XK_Odiaeresis, + //VirtualKeyCode::Multiply => x11::keysym::XK_multiply, + //VirtualKeyCode::Ooblique => x11::keysym::XK_Ooblique, + //VirtualKeyCode::Ugrave => x11::keysym::XK_Ugrave, + //VirtualKeyCode::Uacute => x11::keysym::XK_Uacute, + //VirtualKeyCode::Ucircumflex => x11::keysym::XK_Ucircumflex, + //VirtualKeyCode::Udiaeresis => x11::keysym::XK_Udiaeresis, + //VirtualKeyCode::Yacute => x11::keysym::XK_Yacute, + //VirtualKeyCode::Thorn => x11::keysym::XK_THORN, + //VirtualKeyCode::Thorn => x11::keysym::XK_Thorn, + //VirtualKeyCode::Ssharp => x11::keysym::XK_ssharp, + //VirtualKeyCode::Agrave => x11::keysym::XK_agrave, + //VirtualKeyCode::Aacute => x11::keysym::XK_aacute, + //VirtualKeyCode::Acircumflex => x11::keysym::XK_acircumflex, + //VirtualKeyCode::Atilde => x11::keysym::XK_atilde, + //VirtualKeyCode::Adiaeresis => x11::keysym::XK_adiaeresis, + //VirtualKeyCode::Aring => x11::keysym::XK_aring, + //VirtualKeyCode::Ae => x11::keysym::XK_ae, + //VirtualKeyCode::Ccedilla => x11::keysym::XK_ccedilla, + //VirtualKeyCode::Egrave => x11::keysym::XK_egrave, + //VirtualKeyCode::Eacute => x11::keysym::XK_eacute, + //VirtualKeyCode::Ecircumflex => x11::keysym::XK_ecircumflex, + //VirtualKeyCode::Ediaeresis => x11::keysym::XK_ediaeresis, + //VirtualKeyCode::Igrave => x11::keysym::XK_igrave, + //VirtualKeyCode::Iacute => x11::keysym::XK_iacute, + //VirtualKeyCode::Icircumflex => x11::keysym::XK_icircumflex, + //VirtualKeyCode::Idiaeresis => x11::keysym::XK_idiaeresis, + //VirtualKeyCode::Eth => x11::keysym::XK_eth, + //VirtualKeyCode::Ntilde => x11::keysym::XK_ntilde, + //VirtualKeyCode::Ograve => x11::keysym::XK_ograve, + //VirtualKeyCode::Oacute => x11::keysym::XK_oacute, + //VirtualKeyCode::Ocircumflex => x11::keysym::XK_ocircumflex, + //VirtualKeyCode::Otilde => x11::keysym::XK_otilde, + //VirtualKeyCode::Odiaeresis => x11::keysym::XK_odiaeresis, + //VirtualKeyCode::Division => x11::keysym::XK_division, + //VirtualKeyCode::Oslash => x11::keysym::XK_oslash, + //VirtualKeyCode::Ugrave => x11::keysym::XK_ugrave, + //VirtualKeyCode::Uacute => x11::keysym::XK_uacute, + //VirtualKeyCode::Ucircumflex => x11::keysym::XK_ucircumflex, + //VirtualKeyCode::Udiaeresis => x11::keysym::XK_udiaeresis, + //VirtualKeyCode::Yacute => x11::keysym::XK_yacute, + //VirtualKeyCode::Thorn => x11::keysym::XK_thorn, + //VirtualKeyCode::Ydiaeresis => x11::keysym::XK_ydiaeresis, + //VirtualKeyCode::Aogonek => x11::keysym::XK_Aogonek, + //VirtualKeyCode::Breve => x11::keysym::XK_breve, + //VirtualKeyCode::Lstroke => x11::keysym::XK_Lstroke, + //VirtualKeyCode::Lcaron => x11::keysym::XK_Lcaron, + //VirtualKeyCode::Sacute => x11::keysym::XK_Sacute, + //VirtualKeyCode::Scaron => x11::keysym::XK_Scaron, + //VirtualKeyCode::Scedilla => x11::keysym::XK_Scedilla, + //VirtualKeyCode::Tcaron => x11::keysym::XK_Tcaron, + //VirtualKeyCode::Zacute => x11::keysym::XK_Zacute, + //VirtualKeyCode::Zcaron => x11::keysym::XK_Zcaron, + //VirtualKeyCode::Zabovedot => x11::keysym::XK_Zabovedot, + //VirtualKeyCode::Aogonek => x11::keysym::XK_aogonek, + //VirtualKeyCode::Ogonek => x11::keysym::XK_ogonek, + //VirtualKeyCode::Lstroke => x11::keysym::XK_lstroke, + //VirtualKeyCode::Lcaron => x11::keysym::XK_lcaron, + //VirtualKeyCode::Sacute => x11::keysym::XK_sacute, + //VirtualKeyCode::Caron => x11::keysym::XK_caron, + //VirtualKeyCode::Scaron => x11::keysym::XK_scaron, + //VirtualKeyCode::Scedilla => x11::keysym::XK_scedilla, + //VirtualKeyCode::Tcaron => x11::keysym::XK_tcaron, + //VirtualKeyCode::Zacute => x11::keysym::XK_zacute, + //VirtualKeyCode::Doubleacute => x11::keysym::XK_doubleacute, + //VirtualKeyCode::Zcaron => x11::keysym::XK_zcaron, + //VirtualKeyCode::Zabovedot => x11::keysym::XK_zabovedot, + //VirtualKeyCode::Racute => x11::keysym::XK_Racute, + //VirtualKeyCode::Abreve => x11::keysym::XK_Abreve, + //VirtualKeyCode::Lacute => x11::keysym::XK_Lacute, + //VirtualKeyCode::Cacute => x11::keysym::XK_Cacute, + //VirtualKeyCode::Ccaron => x11::keysym::XK_Ccaron, + //VirtualKeyCode::Eogonek => x11::keysym::XK_Eogonek, + //VirtualKeyCode::Ecaron => x11::keysym::XK_Ecaron, + //VirtualKeyCode::Dcaron => x11::keysym::XK_Dcaron, + //VirtualKeyCode::Dstroke => x11::keysym::XK_Dstroke, + //VirtualKeyCode::Nacute => x11::keysym::XK_Nacute, + //VirtualKeyCode::Ncaron => x11::keysym::XK_Ncaron, + //VirtualKeyCode::Odoubleacute => x11::keysym::XK_Odoubleacute, + //VirtualKeyCode::Rcaron => x11::keysym::XK_Rcaron, + //VirtualKeyCode::Uring => x11::keysym::XK_Uring, + //VirtualKeyCode::Udoubleacute => x11::keysym::XK_Udoubleacute, + //VirtualKeyCode::Tcedilla => x11::keysym::XK_Tcedilla, + //VirtualKeyCode::Racute => x11::keysym::XK_racute, + //VirtualKeyCode::Abreve => x11::keysym::XK_abreve, + //VirtualKeyCode::Lacute => x11::keysym::XK_lacute, + //VirtualKeyCode::Cacute => x11::keysym::XK_cacute, + //VirtualKeyCode::Ccaron => x11::keysym::XK_ccaron, + //VirtualKeyCode::Eogonek => x11::keysym::XK_eogonek, + //VirtualKeyCode::Ecaron => x11::keysym::XK_ecaron, + //VirtualKeyCode::Dcaron => x11::keysym::XK_dcaron, + //VirtualKeyCode::Dstroke => x11::keysym::XK_dstroke, + //VirtualKeyCode::Nacute => x11::keysym::XK_nacute, + //VirtualKeyCode::Ncaron => x11::keysym::XK_ncaron, + //VirtualKeyCode::Odoubleacute => x11::keysym::XK_odoubleacute, + //VirtualKeyCode::Udoubleacute => x11::keysym::XK_udoubleacute, + //VirtualKeyCode::Rcaron => x11::keysym::XK_rcaron, + //VirtualKeyCode::Uring => x11::keysym::XK_uring, + //VirtualKeyCode::Tcedilla => x11::keysym::XK_tcedilla, + //VirtualKeyCode::Abovedot => x11::keysym::XK_abovedot, + //VirtualKeyCode::Hstroke => x11::keysym::XK_Hstroke, + //VirtualKeyCode::Hcircumflex => x11::keysym::XK_Hcircumflex, + //VirtualKeyCode::Iabovedot => x11::keysym::XK_Iabovedot, + //VirtualKeyCode::Gbreve => x11::keysym::XK_Gbreve, + //VirtualKeyCode::Jcircumflex => x11::keysym::XK_Jcircumflex, + //VirtualKeyCode::Hstroke => x11::keysym::XK_hstroke, + //VirtualKeyCode::Hcircumflex => x11::keysym::XK_hcircumflex, + //VirtualKeyCode::Idotless => x11::keysym::XK_idotless, + //VirtualKeyCode::Gbreve => x11::keysym::XK_gbreve, + //VirtualKeyCode::Jcircumflex => x11::keysym::XK_jcircumflex, + //VirtualKeyCode::Cabovedot => x11::keysym::XK_Cabovedot, + //VirtualKeyCode::Ccircumflex => x11::keysym::XK_Ccircumflex, + //VirtualKeyCode::Gabovedot => x11::keysym::XK_Gabovedot, + //VirtualKeyCode::Gcircumflex => x11::keysym::XK_Gcircumflex, + //VirtualKeyCode::Ubreve => x11::keysym::XK_Ubreve, + //VirtualKeyCode::Scircumflex => x11::keysym::XK_Scircumflex, + //VirtualKeyCode::Cabovedot => x11::keysym::XK_cabovedot, + //VirtualKeyCode::Ccircumflex => x11::keysym::XK_ccircumflex, + //VirtualKeyCode::Gabovedot => x11::keysym::XK_gabovedot, + //VirtualKeyCode::Gcircumflex => x11::keysym::XK_gcircumflex, + //VirtualKeyCode::Ubreve => x11::keysym::XK_ubreve, + //VirtualKeyCode::Scircumflex => x11::keysym::XK_scircumflex, + //VirtualKeyCode::Kra => x11::keysym::XK_kra, + //VirtualKeyCode::Kappa => x11::keysym::XK_kappa, + //VirtualKeyCode::Rcedilla => x11::keysym::XK_Rcedilla, + //VirtualKeyCode::Itilde => x11::keysym::XK_Itilde, + //VirtualKeyCode::Lcedilla => x11::keysym::XK_Lcedilla, + //VirtualKeyCode::Emacron => x11::keysym::XK_Emacron, + //VirtualKeyCode::Gcedilla => x11::keysym::XK_Gcedilla, + //VirtualKeyCode::Tslash => x11::keysym::XK_Tslash, + //VirtualKeyCode::Rcedilla => x11::keysym::XK_rcedilla, + //VirtualKeyCode::Itilde => x11::keysym::XK_itilde, + //VirtualKeyCode::Lcedilla => x11::keysym::XK_lcedilla, + //VirtualKeyCode::Emacron => x11::keysym::XK_emacron, + //VirtualKeyCode::Gcedilla => x11::keysym::XK_gcedilla, + //VirtualKeyCode::Tslash => x11::keysym::XK_tslash, + //VirtualKeyCode::Eng => x11::keysym::XK_ENG, + //VirtualKeyCode::Eng => x11::keysym::XK_eng, + //VirtualKeyCode::Amacron => x11::keysym::XK_Amacron, + //VirtualKeyCode::Iogonek => x11::keysym::XK_Iogonek, + //VirtualKeyCode::Eabovedot => x11::keysym::XK_Eabovedot, + //VirtualKeyCode::Imacron => x11::keysym::XK_Imacron, + //VirtualKeyCode::Ncedilla => x11::keysym::XK_Ncedilla, + //VirtualKeyCode::Omacron => x11::keysym::XK_Omacron, + //VirtualKeyCode::Kcedilla => x11::keysym::XK_Kcedilla, + //VirtualKeyCode::Uogonek => x11::keysym::XK_Uogonek, + //VirtualKeyCode::Utilde => x11::keysym::XK_Utilde, + //VirtualKeyCode::Umacron => x11::keysym::XK_Umacron, + //VirtualKeyCode::Amacron => x11::keysym::XK_amacron, + //VirtualKeyCode::Iogonek => x11::keysym::XK_iogonek, + //VirtualKeyCode::Eabovedot => x11::keysym::XK_eabovedot, + //VirtualKeyCode::Imacron => x11::keysym::XK_imacron, + //VirtualKeyCode::Ncedilla => x11::keysym::XK_ncedilla, + //VirtualKeyCode::Omacron => x11::keysym::XK_omacron, + //VirtualKeyCode::Kcedilla => x11::keysym::XK_kcedilla, + //VirtualKeyCode::Uogonek => x11::keysym::XK_uogonek, + //VirtualKeyCode::Utilde => x11::keysym::XK_utilde, + //VirtualKeyCode::Umacron => x11::keysym::XK_umacron, + //VirtualKeyCode::Overline => x11::keysym::XK_overline, + //VirtualKeyCode::Kana_fullstop => x11::keysym::XK_kana_fullstop, + //VirtualKeyCode::Kana_openingbracket => x11::keysym::XK_kana_openingbracket, + //VirtualKeyCode::Kana_closingbracket => x11::keysym::XK_kana_closingbracket, + //VirtualKeyCode::Kana_comma => x11::keysym::XK_kana_comma, + //VirtualKeyCode::Kana_conjunctive => x11::keysym::XK_kana_conjunctive, + //VirtualKeyCode::Kana_middledot => x11::keysym::XK_kana_middledot, + //VirtualKeyCode::Kana_wo => x11::keysym::XK_kana_WO, + //VirtualKeyCode::Kana_a => x11::keysym::XK_kana_a, + //VirtualKeyCode::Kana_i => x11::keysym::XK_kana_i, + //VirtualKeyCode::Kana_u => x11::keysym::XK_kana_u, + //VirtualKeyCode::Kana_e => x11::keysym::XK_kana_e, + //VirtualKeyCode::Kana_o => x11::keysym::XK_kana_o, + //VirtualKeyCode::Kana_ya => x11::keysym::XK_kana_ya, + //VirtualKeyCode::Kana_yu => x11::keysym::XK_kana_yu, + //VirtualKeyCode::Kana_yo => x11::keysym::XK_kana_yo, + //VirtualKeyCode::Kana_tsu => x11::keysym::XK_kana_tsu, + //VirtualKeyCode::Kana_tu => x11::keysym::XK_kana_tu, + //VirtualKeyCode::Prolongedsound => x11::keysym::XK_prolongedsound, + //VirtualKeyCode::Kana_a => x11::keysym::XK_kana_A, + //VirtualKeyCode::Kana_i => x11::keysym::XK_kana_I, + //VirtualKeyCode::Kana_u => x11::keysym::XK_kana_U, + //VirtualKeyCode::Kana_e => x11::keysym::XK_kana_E, + //VirtualKeyCode::Kana_o => x11::keysym::XK_kana_O, + //VirtualKeyCode::Kana_ka => x11::keysym::XK_kana_KA, + //VirtualKeyCode::Kana_ki => x11::keysym::XK_kana_KI, + //VirtualKeyCode::Kana_ku => x11::keysym::XK_kana_KU, + //VirtualKeyCode::Kana_ke => x11::keysym::XK_kana_KE, + //VirtualKeyCode::Kana_ko => x11::keysym::XK_kana_KO, + //VirtualKeyCode::Kana_sa => x11::keysym::XK_kana_SA, + //VirtualKeyCode::Kana_shi => x11::keysym::XK_kana_SHI, + //VirtualKeyCode::Kana_su => x11::keysym::XK_kana_SU, + //VirtualKeyCode::Kana_se => x11::keysym::XK_kana_SE, + //VirtualKeyCode::Kana_so => x11::keysym::XK_kana_SO, + //VirtualKeyCode::Kana_ta => x11::keysym::XK_kana_TA, + //VirtualKeyCode::Kana_chi => x11::keysym::XK_kana_CHI, + //VirtualKeyCode::Kana_ti => x11::keysym::XK_kana_TI, + //VirtualKeyCode::Kana_tsu => x11::keysym::XK_kana_TSU, + //VirtualKeyCode::Kana_tu => x11::keysym::XK_kana_TU, + //VirtualKeyCode::Kana_te => x11::keysym::XK_kana_TE, + //VirtualKeyCode::Kana_to => x11::keysym::XK_kana_TO, + //VirtualKeyCode::Kana_na => x11::keysym::XK_kana_NA, + //VirtualKeyCode::Kana_ni => x11::keysym::XK_kana_NI, + //VirtualKeyCode::Kana_nu => x11::keysym::XK_kana_NU, + //VirtualKeyCode::Kana_ne => x11::keysym::XK_kana_NE, + //VirtualKeyCode::Kana_no => x11::keysym::XK_kana_NO, + //VirtualKeyCode::Kana_ha => x11::keysym::XK_kana_HA, + //VirtualKeyCode::Kana_hi => x11::keysym::XK_kana_HI, + //VirtualKeyCode::Kana_fu => x11::keysym::XK_kana_FU, + //VirtualKeyCode::Kana_hu => x11::keysym::XK_kana_HU, + //VirtualKeyCode::Kana_he => x11::keysym::XK_kana_HE, + //VirtualKeyCode::Kana_ho => x11::keysym::XK_kana_HO, + //VirtualKeyCode::Kana_ma => x11::keysym::XK_kana_MA, + //VirtualKeyCode::Kana_mi => x11::keysym::XK_kana_MI, + //VirtualKeyCode::Kana_mu => x11::keysym::XK_kana_MU, + //VirtualKeyCode::Kana_me => x11::keysym::XK_kana_ME, + //VirtualKeyCode::Kana_mo => x11::keysym::XK_kana_MO, + //VirtualKeyCode::Kana_ya => x11::keysym::XK_kana_YA, + //VirtualKeyCode::Kana_yu => x11::keysym::XK_kana_YU, + //VirtualKeyCode::Kana_yo => x11::keysym::XK_kana_YO, + //VirtualKeyCode::Kana_ra => x11::keysym::XK_kana_RA, + //VirtualKeyCode::Kana_ri => x11::keysym::XK_kana_RI, + //VirtualKeyCode::Kana_ru => x11::keysym::XK_kana_RU, + //VirtualKeyCode::Kana_re => x11::keysym::XK_kana_RE, + //VirtualKeyCode::Kana_ro => x11::keysym::XK_kana_RO, + //VirtualKeyCode::Kana_wa => x11::keysym::XK_kana_WA, + //VirtualKeyCode::Kana_n => x11::keysym::XK_kana_N, + //VirtualKeyCode::Voicedsound => x11::keysym::XK_voicedsound, + //VirtualKeyCode::Semivoicedsound => x11::keysym::XK_semivoicedsound, + //VirtualKeyCode::Kana_switch => x11::keysym::XK_kana_switch, + //VirtualKeyCode::Arabic_comma => x11::keysym::XK_Arabic_comma, + //VirtualKeyCode::Arabic_semicolon => x11::keysym::XK_Arabic_semicolon, + //VirtualKeyCode::Arabic_question_mark => x11::keysym::XK_Arabic_question_mark, + //VirtualKeyCode::Arabic_hamza => x11::keysym::XK_Arabic_hamza, + //VirtualKeyCode::Arabic_maddaonalef => x11::keysym::XK_Arabic_maddaonalef, + //VirtualKeyCode::Arabic_hamzaonalef => x11::keysym::XK_Arabic_hamzaonalef, + //VirtualKeyCode::Arabic_hamzaonwaw => x11::keysym::XK_Arabic_hamzaonwaw, + //VirtualKeyCode::Arabic_hamzaunderalef => x11::keysym::XK_Arabic_hamzaunderalef, + //VirtualKeyCode::Arabic_hamzaonyeh => x11::keysym::XK_Arabic_hamzaonyeh, + //VirtualKeyCode::Arabic_alef => x11::keysym::XK_Arabic_alef, + //VirtualKeyCode::Arabic_beh => x11::keysym::XK_Arabic_beh, + //VirtualKeyCode::Arabic_tehmarbuta => x11::keysym::XK_Arabic_tehmarbuta, + //VirtualKeyCode::Arabic_teh => x11::keysym::XK_Arabic_teh, + //VirtualKeyCode::Arabic_theh => x11::keysym::XK_Arabic_theh, + //VirtualKeyCode::Arabic_jeem => x11::keysym::XK_Arabic_jeem, + //VirtualKeyCode::Arabic_hah => x11::keysym::XK_Arabic_hah, + //VirtualKeyCode::Arabic_khah => x11::keysym::XK_Arabic_khah, + //VirtualKeyCode::Arabic_dal => x11::keysym::XK_Arabic_dal, + //VirtualKeyCode::Arabic_thal => x11::keysym::XK_Arabic_thal, + //VirtualKeyCode::Arabic_ra => x11::keysym::XK_Arabic_ra, + //VirtualKeyCode::Arabic_zain => x11::keysym::XK_Arabic_zain, + //VirtualKeyCode::Arabic_seen => x11::keysym::XK_Arabic_seen, + //VirtualKeyCode::Arabic_sheen => x11::keysym::XK_Arabic_sheen, + //VirtualKeyCode::Arabic_sad => x11::keysym::XK_Arabic_sad, + //VirtualKeyCode::Arabic_dad => x11::keysym::XK_Arabic_dad, + //VirtualKeyCode::Arabic_tah => x11::keysym::XK_Arabic_tah, + //VirtualKeyCode::Arabic_zah => x11::keysym::XK_Arabic_zah, + //VirtualKeyCode::Arabic_ain => x11::keysym::XK_Arabic_ain, + //VirtualKeyCode::Arabic_ghain => x11::keysym::XK_Arabic_ghain, + //VirtualKeyCode::Arabic_tatweel => x11::keysym::XK_Arabic_tatweel, + //VirtualKeyCode::Arabic_feh => x11::keysym::XK_Arabic_feh, + //VirtualKeyCode::Arabic_qaf => x11::keysym::XK_Arabic_qaf, + //VirtualKeyCode::Arabic_kaf => x11::keysym::XK_Arabic_kaf, + //VirtualKeyCode::Arabic_lam => x11::keysym::XK_Arabic_lam, + //VirtualKeyCode::Arabic_meem => x11::keysym::XK_Arabic_meem, + //VirtualKeyCode::Arabic_noon => x11::keysym::XK_Arabic_noon, + //VirtualKeyCode::Arabic_ha => x11::keysym::XK_Arabic_ha, + //VirtualKeyCode::Arabic_heh => x11::keysym::XK_Arabic_heh, + //VirtualKeyCode::Arabic_waw => x11::keysym::XK_Arabic_waw, + //VirtualKeyCode::Arabic_alefmaksura => x11::keysym::XK_Arabic_alefmaksura, + //VirtualKeyCode::Arabic_yeh => x11::keysym::XK_Arabic_yeh, + //VirtualKeyCode::Arabic_fathatan => x11::keysym::XK_Arabic_fathatan, + //VirtualKeyCode::Arabic_dammatan => x11::keysym::XK_Arabic_dammatan, + //VirtualKeyCode::Arabic_kasratan => x11::keysym::XK_Arabic_kasratan, + //VirtualKeyCode::Arabic_fatha => x11::keysym::XK_Arabic_fatha, + //VirtualKeyCode::Arabic_damma => x11::keysym::XK_Arabic_damma, + //VirtualKeyCode::Arabic_kasra => x11::keysym::XK_Arabic_kasra, + //VirtualKeyCode::Arabic_shadda => x11::keysym::XK_Arabic_shadda, + //VirtualKeyCode::Arabic_sukun => x11::keysym::XK_Arabic_sukun, + //VirtualKeyCode::Arabic_switch => x11::keysym::XK_Arabic_switch, + //VirtualKeyCode::Serbian_dje => x11::keysym::XK_Serbian_dje, + //VirtualKeyCode::Macedonia_gje => x11::keysym::XK_Macedonia_gje, + //VirtualKeyCode::Cyrillic_io => x11::keysym::XK_Cyrillic_io, + //VirtualKeyCode::Ukrainian_ie => x11::keysym::XK_Ukrainian_ie, + //VirtualKeyCode::Ukranian_je => x11::keysym::XK_Ukranian_je, + //VirtualKeyCode::Macedonia_dse => x11::keysym::XK_Macedonia_dse, + //VirtualKeyCode::Ukrainian_i => x11::keysym::XK_Ukrainian_i, + //VirtualKeyCode::Ukranian_i => x11::keysym::XK_Ukranian_i, + //VirtualKeyCode::Ukrainian_yi => x11::keysym::XK_Ukrainian_yi, + //VirtualKeyCode::Ukranian_yi => x11::keysym::XK_Ukranian_yi, + //VirtualKeyCode::Cyrillic_je => x11::keysym::XK_Cyrillic_je, + //VirtualKeyCode::Serbian_je => x11::keysym::XK_Serbian_je, + //VirtualKeyCode::Cyrillic_lje => x11::keysym::XK_Cyrillic_lje, + //VirtualKeyCode::Serbian_lje => x11::keysym::XK_Serbian_lje, + //VirtualKeyCode::Cyrillic_nje => x11::keysym::XK_Cyrillic_nje, + //VirtualKeyCode::Serbian_nje => x11::keysym::XK_Serbian_nje, + //VirtualKeyCode::Serbian_tshe => x11::keysym::XK_Serbian_tshe, + //VirtualKeyCode::Macedonia_kje => x11::keysym::XK_Macedonia_kje, + //VirtualKeyCode::Byelorussian_shortu => x11::keysym::XK_Byelorussian_shortu, + //VirtualKeyCode::Cyrillic_dzhe => x11::keysym::XK_Cyrillic_dzhe, + //VirtualKeyCode::Serbian_dze => x11::keysym::XK_Serbian_dze, + //VirtualKeyCode::Numerosign => x11::keysym::XK_numerosign, + //VirtualKeyCode::Serbian_dje => x11::keysym::XK_Serbian_DJE, + //VirtualKeyCode::Macedonia_gje => x11::keysym::XK_Macedonia_GJE, + //VirtualKeyCode::Cyrillic_io => x11::keysym::XK_Cyrillic_IO, + //VirtualKeyCode::Ukrainian_ie => x11::keysym::XK_Ukrainian_IE, + //VirtualKeyCode::Ukranian_je => x11::keysym::XK_Ukranian_JE, + //VirtualKeyCode::Macedonia_dse => x11::keysym::XK_Macedonia_DSE, + //VirtualKeyCode::Ukrainian_i => x11::keysym::XK_Ukrainian_I, + //VirtualKeyCode::Ukranian_i => x11::keysym::XK_Ukranian_I, + //VirtualKeyCode::Ukrainian_yi => x11::keysym::XK_Ukrainian_YI, + //VirtualKeyCode::Ukranian_yi => x11::keysym::XK_Ukranian_YI, + //VirtualKeyCode::Cyrillic_je => x11::keysym::XK_Cyrillic_JE, + //VirtualKeyCode::Serbian_je => x11::keysym::XK_Serbian_JE, + //VirtualKeyCode::Cyrillic_lje => x11::keysym::XK_Cyrillic_LJE, + //VirtualKeyCode::Serbian_lje => x11::keysym::XK_Serbian_LJE, + //VirtualKeyCode::Cyrillic_nje => x11::keysym::XK_Cyrillic_NJE, + //VirtualKeyCode::Serbian_nje => x11::keysym::XK_Serbian_NJE, + //VirtualKeyCode::Serbian_tshe => x11::keysym::XK_Serbian_TSHE, + //VirtualKeyCode::Macedonia_kje => x11::keysym::XK_Macedonia_KJE, + //VirtualKeyCode::Byelorussian_shortu => x11::keysym::XK_Byelorussian_SHORTU, + //VirtualKeyCode::Cyrillic_dzhe => x11::keysym::XK_Cyrillic_DZHE, + //VirtualKeyCode::Serbian_dze => x11::keysym::XK_Serbian_DZE, + //VirtualKeyCode::Cyrillic_yu => x11::keysym::XK_Cyrillic_yu, + //VirtualKeyCode::Cyrillic_a => x11::keysym::XK_Cyrillic_a, + //VirtualKeyCode::Cyrillic_be => x11::keysym::XK_Cyrillic_be, + //VirtualKeyCode::Cyrillic_tse => x11::keysym::XK_Cyrillic_tse, + //VirtualKeyCode::Cyrillic_de => x11::keysym::XK_Cyrillic_de, + //VirtualKeyCode::Cyrillic_ie => x11::keysym::XK_Cyrillic_ie, + //VirtualKeyCode::Cyrillic_ef => x11::keysym::XK_Cyrillic_ef, + //VirtualKeyCode::Cyrillic_ghe => x11::keysym::XK_Cyrillic_ghe, + //VirtualKeyCode::Cyrillic_ha => x11::keysym::XK_Cyrillic_ha, + //VirtualKeyCode::Cyrillic_i => x11::keysym::XK_Cyrillic_i, + //VirtualKeyCode::Cyrillic_shorti => x11::keysym::XK_Cyrillic_shorti, + //VirtualKeyCode::Cyrillic_ka => x11::keysym::XK_Cyrillic_ka, + //VirtualKeyCode::Cyrillic_el => x11::keysym::XK_Cyrillic_el, + //VirtualKeyCode::Cyrillic_em => x11::keysym::XK_Cyrillic_em, + //VirtualKeyCode::Cyrillic_en => x11::keysym::XK_Cyrillic_en, + //VirtualKeyCode::Cyrillic_o => x11::keysym::XK_Cyrillic_o, + //VirtualKeyCode::Cyrillic_pe => x11::keysym::XK_Cyrillic_pe, + //VirtualKeyCode::Cyrillic_ya => x11::keysym::XK_Cyrillic_ya, + //VirtualKeyCode::Cyrillic_er => x11::keysym::XK_Cyrillic_er, + //VirtualKeyCode::Cyrillic_es => x11::keysym::XK_Cyrillic_es, + //VirtualKeyCode::Cyrillic_te => x11::keysym::XK_Cyrillic_te, + //VirtualKeyCode::Cyrillic_u => x11::keysym::XK_Cyrillic_u, + //VirtualKeyCode::Cyrillic_zhe => x11::keysym::XK_Cyrillic_zhe, + //VirtualKeyCode::Cyrillic_ve => x11::keysym::XK_Cyrillic_ve, + //VirtualKeyCode::Cyrillic_softsign => x11::keysym::XK_Cyrillic_softsign, + //VirtualKeyCode::Cyrillic_yeru => x11::keysym::XK_Cyrillic_yeru, + //VirtualKeyCode::Cyrillic_ze => x11::keysym::XK_Cyrillic_ze, + //VirtualKeyCode::Cyrillic_sha => x11::keysym::XK_Cyrillic_sha, + //VirtualKeyCode::Cyrillic_e => x11::keysym::XK_Cyrillic_e, + //VirtualKeyCode::Cyrillic_shcha => x11::keysym::XK_Cyrillic_shcha, + //VirtualKeyCode::Cyrillic_che => x11::keysym::XK_Cyrillic_che, + //VirtualKeyCode::Cyrillic_hardsign => x11::keysym::XK_Cyrillic_hardsign, + //VirtualKeyCode::Cyrillic_yu => x11::keysym::XK_Cyrillic_YU, + //VirtualKeyCode::Cyrillic_a => x11::keysym::XK_Cyrillic_A, + //VirtualKeyCode::Cyrillic_be => x11::keysym::XK_Cyrillic_BE, + //VirtualKeyCode::Cyrillic_tse => x11::keysym::XK_Cyrillic_TSE, + //VirtualKeyCode::Cyrillic_de => x11::keysym::XK_Cyrillic_DE, + //VirtualKeyCode::Cyrillic_ie => x11::keysym::XK_Cyrillic_IE, + //VirtualKeyCode::Cyrillic_ef => x11::keysym::XK_Cyrillic_EF, + //VirtualKeyCode::Cyrillic_ghe => x11::keysym::XK_Cyrillic_GHE, + //VirtualKeyCode::Cyrillic_ha => x11::keysym::XK_Cyrillic_HA, + //VirtualKeyCode::Cyrillic_i => x11::keysym::XK_Cyrillic_I, + //VirtualKeyCode::Cyrillic_shorti => x11::keysym::XK_Cyrillic_SHORTI, + //VirtualKeyCode::Cyrillic_ka => x11::keysym::XK_Cyrillic_KA, + //VirtualKeyCode::Cyrillic_el => x11::keysym::XK_Cyrillic_EL, + //VirtualKeyCode::Cyrillic_em => x11::keysym::XK_Cyrillic_EM, + //VirtualKeyCode::Cyrillic_en => x11::keysym::XK_Cyrillic_EN, + //VirtualKeyCode::Cyrillic_o => x11::keysym::XK_Cyrillic_O, + //VirtualKeyCode::Cyrillic_pe => x11::keysym::XK_Cyrillic_PE, + //VirtualKeyCode::Cyrillic_ya => x11::keysym::XK_Cyrillic_YA, + //VirtualKeyCode::Cyrillic_er => x11::keysym::XK_Cyrillic_ER, + //VirtualKeyCode::Cyrillic_es => x11::keysym::XK_Cyrillic_ES, + //VirtualKeyCode::Cyrillic_te => x11::keysym::XK_Cyrillic_TE, + //VirtualKeyCode::Cyrillic_u => x11::keysym::XK_Cyrillic_U, + //VirtualKeyCode::Cyrillic_zhe => x11::keysym::XK_Cyrillic_ZHE, + //VirtualKeyCode::Cyrillic_ve => x11::keysym::XK_Cyrillic_VE, + //VirtualKeyCode::Cyrillic_softsign => x11::keysym::XK_Cyrillic_SOFTSIGN, + //VirtualKeyCode::Cyrillic_yeru => x11::keysym::XK_Cyrillic_YERU, + //VirtualKeyCode::Cyrillic_ze => x11::keysym::XK_Cyrillic_ZE, + //VirtualKeyCode::Cyrillic_sha => x11::keysym::XK_Cyrillic_SHA, + //VirtualKeyCode::Cyrillic_e => x11::keysym::XK_Cyrillic_E, + //VirtualKeyCode::Cyrillic_shcha => x11::keysym::XK_Cyrillic_SHCHA, + //VirtualKeyCode::Cyrillic_che => x11::keysym::XK_Cyrillic_CHE, + //VirtualKeyCode::Cyrillic_hardsign => x11::keysym::XK_Cyrillic_HARDSIGN, + //VirtualKeyCode::Greek_alphaaccent => x11::keysym::XK_Greek_ALPHAaccent, + //VirtualKeyCode::Greek_epsilonaccent => x11::keysym::XK_Greek_EPSILONaccent, + //VirtualKeyCode::Greek_etaaccent => x11::keysym::XK_Greek_ETAaccent, + //VirtualKeyCode::Greek_iotaaccent => x11::keysym::XK_Greek_IOTAaccent, + //VirtualKeyCode::Greek_iotadiaeresis => x11::keysym::XK_Greek_IOTAdiaeresis, + //VirtualKeyCode::Greek_omicronaccent => x11::keysym::XK_Greek_OMICRONaccent, + //VirtualKeyCode::Greek_upsilonaccent => x11::keysym::XK_Greek_UPSILONaccent, + //VirtualKeyCode::Greek_upsilondieresis => x11::keysym::XK_Greek_UPSILONdieresis, + //VirtualKeyCode::Greek_omegaaccent => x11::keysym::XK_Greek_OMEGAaccent, + //VirtualKeyCode::Greek_accentdieresis => x11::keysym::XK_Greek_accentdieresis, + //VirtualKeyCode::Greek_horizbar => x11::keysym::XK_Greek_horizbar, + //VirtualKeyCode::Greek_alphaaccent => x11::keysym::XK_Greek_alphaaccent, + //VirtualKeyCode::Greek_epsilonaccent => x11::keysym::XK_Greek_epsilonaccent, + //VirtualKeyCode::Greek_etaaccent => x11::keysym::XK_Greek_etaaccent, + //VirtualKeyCode::Greek_iotaaccent => x11::keysym::XK_Greek_iotaaccent, + //VirtualKeyCode::Greek_iotadieresis => x11::keysym::XK_Greek_iotadieresis, + //VirtualKeyCode::Greek_iotaaccentdieresis => x11::keysym::XK_Greek_iotaaccentdieresis, + //VirtualKeyCode::Greek_omicronaccent => x11::keysym::XK_Greek_omicronaccent, + //VirtualKeyCode::Greek_upsilonaccent => x11::keysym::XK_Greek_upsilonaccent, + //VirtualKeyCode::Greek_upsilondieresis => x11::keysym::XK_Greek_upsilondieresis, + //VirtualKeyCode::Greek_upsilonaccentdieresis => x11::keysym::XK_Greek_upsilonaccentdieresis, + //VirtualKeyCode::Greek_omegaaccent => x11::keysym::XK_Greek_omegaaccent, + //VirtualKeyCode::Greek_alpha => x11::keysym::XK_Greek_ALPHA, + //VirtualKeyCode::Greek_beta => x11::keysym::XK_Greek_BETA, + //VirtualKeyCode::Greek_gamma => x11::keysym::XK_Greek_GAMMA, + //VirtualKeyCode::Greek_delta => x11::keysym::XK_Greek_DELTA, + //VirtualKeyCode::Greek_epsilon => x11::keysym::XK_Greek_EPSILON, + //VirtualKeyCode::Greek_zeta => x11::keysym::XK_Greek_ZETA, + //VirtualKeyCode::Greek_eta => x11::keysym::XK_Greek_ETA, + //VirtualKeyCode::Greek_theta => x11::keysym::XK_Greek_THETA, + //VirtualKeyCode::Greek_iota => x11::keysym::XK_Greek_IOTA, + //VirtualKeyCode::Greek_kappa => x11::keysym::XK_Greek_KAPPA, + //VirtualKeyCode::Greek_lamda => x11::keysym::XK_Greek_LAMDA, + //VirtualKeyCode::Greek_lambda => x11::keysym::XK_Greek_LAMBDA, + //VirtualKeyCode::Greek_mu => x11::keysym::XK_Greek_MU, + //VirtualKeyCode::Greek_nu => x11::keysym::XK_Greek_NU, + //VirtualKeyCode::Greek_xi => x11::keysym::XK_Greek_XI, + //VirtualKeyCode::Greek_omicron => x11::keysym::XK_Greek_OMICRON, + //VirtualKeyCode::Greek_pi => x11::keysym::XK_Greek_PI, + //VirtualKeyCode::Greek_rho => x11::keysym::XK_Greek_RHO, + //VirtualKeyCode::Greek_sigma => x11::keysym::XK_Greek_SIGMA, + //VirtualKeyCode::Greek_tau => x11::keysym::XK_Greek_TAU, + //VirtualKeyCode::Greek_upsilon => x11::keysym::XK_Greek_UPSILON, + //VirtualKeyCode::Greek_phi => x11::keysym::XK_Greek_PHI, + //VirtualKeyCode::Greek_chi => x11::keysym::XK_Greek_CHI, + //VirtualKeyCode::Greek_psi => x11::keysym::XK_Greek_PSI, + //VirtualKeyCode::Greek_omega => x11::keysym::XK_Greek_OMEGA, + //VirtualKeyCode::Greek_alpha => x11::keysym::XK_Greek_alpha, + //VirtualKeyCode::Greek_beta => x11::keysym::XK_Greek_beta, + //VirtualKeyCode::Greek_gamma => x11::keysym::XK_Greek_gamma, + //VirtualKeyCode::Greek_delta => x11::keysym::XK_Greek_delta, + //VirtualKeyCode::Greek_epsilon => x11::keysym::XK_Greek_epsilon, + //VirtualKeyCode::Greek_zeta => x11::keysym::XK_Greek_zeta, + //VirtualKeyCode::Greek_eta => x11::keysym::XK_Greek_eta, + //VirtualKeyCode::Greek_theta => x11::keysym::XK_Greek_theta, + //VirtualKeyCode::Greek_iota => x11::keysym::XK_Greek_iota, + //VirtualKeyCode::Greek_kappa => x11::keysym::XK_Greek_kappa, + //VirtualKeyCode::Greek_lamda => x11::keysym::XK_Greek_lamda, + //VirtualKeyCode::Greek_lambda => x11::keysym::XK_Greek_lambda, + //VirtualKeyCode::Greek_mu => x11::keysym::XK_Greek_mu, + //VirtualKeyCode::Greek_nu => x11::keysym::XK_Greek_nu, + //VirtualKeyCode::Greek_xi => x11::keysym::XK_Greek_xi, + //VirtualKeyCode::Greek_omicron => x11::keysym::XK_Greek_omicron, + //VirtualKeyCode::Greek_pi => x11::keysym::XK_Greek_pi, + //VirtualKeyCode::Greek_rho => x11::keysym::XK_Greek_rho, + //VirtualKeyCode::Greek_sigma => x11::keysym::XK_Greek_sigma, + //VirtualKeyCode::Greek_finalsmallsigma => x11::keysym::XK_Greek_finalsmallsigma, + //VirtualKeyCode::Greek_tau => x11::keysym::XK_Greek_tau, + //VirtualKeyCode::Greek_upsilon => x11::keysym::XK_Greek_upsilon, + //VirtualKeyCode::Greek_phi => x11::keysym::XK_Greek_phi, + //VirtualKeyCode::Greek_chi => x11::keysym::XK_Greek_chi, + //VirtualKeyCode::Greek_psi => x11::keysym::XK_Greek_psi, + //VirtualKeyCode::Greek_omega => x11::keysym::XK_Greek_omega, + //VirtualKeyCode::Greek_switch => x11::keysym::XK_Greek_switch, + //VirtualKeyCode::Leftradical => x11::keysym::XK_leftradical, + //VirtualKeyCode::Topleftradical => x11::keysym::XK_topleftradical, + //VirtualKeyCode::Horizconnector => x11::keysym::XK_horizconnector, + //VirtualKeyCode::Topintegral => x11::keysym::XK_topintegral, + //VirtualKeyCode::Botintegral => x11::keysym::XK_botintegral, + //VirtualKeyCode::Vertconnector => x11::keysym::XK_vertconnector, + //VirtualKeyCode::Topleftsqbracket => x11::keysym::XK_topleftsqbracket, + //VirtualKeyCode::Botleftsqbracket => x11::keysym::XK_botleftsqbracket, + //VirtualKeyCode::Toprightsqbracket => x11::keysym::XK_toprightsqbracket, + //VirtualKeyCode::Botrightsqbracket => x11::keysym::XK_botrightsqbracket, + //VirtualKeyCode::Topleftparens => x11::keysym::XK_topleftparens, + //VirtualKeyCode::Botleftparens => x11::keysym::XK_botleftparens, + //VirtualKeyCode::Toprightparens => x11::keysym::XK_toprightparens, + //VirtualKeyCode::Botrightparens => x11::keysym::XK_botrightparens, + //VirtualKeyCode::Leftmiddlecurlybrace => x11::keysym::XK_leftmiddlecurlybrace, + //VirtualKeyCode::Rightmiddlecurlybrace => x11::keysym::XK_rightmiddlecurlybrace, + //VirtualKeyCode::Topleftsummation => x11::keysym::XK_topleftsummation, + //VirtualKeyCode::Botleftsummation => x11::keysym::XK_botleftsummation, + //VirtualKeyCode::Topvertsummationconnector => x11::keysym::XK_topvertsummationconnector, + //VirtualKeyCode::Botvertsummationconnector => x11::keysym::XK_botvertsummationconnector, + //VirtualKeyCode::Toprightsummation => x11::keysym::XK_toprightsummation, + //VirtualKeyCode::Botrightsummation => x11::keysym::XK_botrightsummation, + //VirtualKeyCode::Rightmiddlesummation => x11::keysym::XK_rightmiddlesummation, + //VirtualKeyCode::Lessthanequal => x11::keysym::XK_lessthanequal, + //VirtualKeyCode::Notequal => x11::keysym::XK_notequal, + //VirtualKeyCode::Greaterthanequal => x11::keysym::XK_greaterthanequal, + //VirtualKeyCode::Integral => x11::keysym::XK_integral, + //VirtualKeyCode::Therefore => x11::keysym::XK_therefore, + //VirtualKeyCode::Variation => x11::keysym::XK_variation, + //VirtualKeyCode::Infinity => x11::keysym::XK_infinity, + //VirtualKeyCode::Nabla => x11::keysym::XK_nabla, + //VirtualKeyCode::Approximate => x11::keysym::XK_approximate, + //VirtualKeyCode::Similarequal => x11::keysym::XK_similarequal, + //VirtualKeyCode::Ifonlyif => x11::keysym::XK_ifonlyif, + //VirtualKeyCode::Implies => x11::keysym::XK_implies, + //VirtualKeyCode::Identical => x11::keysym::XK_identical, + //VirtualKeyCode::Radical => x11::keysym::XK_radical, + //VirtualKeyCode::Includedin => x11::keysym::XK_includedin, + //VirtualKeyCode::Includes => x11::keysym::XK_includes, + //VirtualKeyCode::Intersection => x11::keysym::XK_intersection, + //VirtualKeyCode::Union => x11::keysym::XK_union, + //VirtualKeyCode::Logicaland => x11::keysym::XK_logicaland, + //VirtualKeyCode::Logicalor => x11::keysym::XK_logicalor, + //VirtualKeyCode::Partialderivative => x11::keysym::XK_partialderivative, + //VirtualKeyCode::Function => x11::keysym::XK_function, + //VirtualKeyCode::Leftarrow => x11::keysym::XK_leftarrow, + //VirtualKeyCode::Uparrow => x11::keysym::XK_uparrow, + //VirtualKeyCode::Rightarrow => x11::keysym::XK_rightarrow, + //VirtualKeyCode::Downarrow => x11::keysym::XK_downarrow, + //VirtualKeyCode::Blank => x11::keysym::XK_blank, + //VirtualKeyCode::Soliddiamond => x11::keysym::XK_soliddiamond, + //VirtualKeyCode::Checkerboard => x11::keysym::XK_checkerboard, + //VirtualKeyCode::Ht => x11::keysym::XK_ht, + //VirtualKeyCode::Ff => x11::keysym::XK_ff, + //VirtualKeyCode::Cr => x11::keysym::XK_cr, + //VirtualKeyCode::Lf => x11::keysym::XK_lf, + //VirtualKeyCode::Nl => x11::keysym::XK_nl, + //VirtualKeyCode::Vt => x11::keysym::XK_vt, + //VirtualKeyCode::Lowrightcorner => x11::keysym::XK_lowrightcorner, + //VirtualKeyCode::Uprightcorner => x11::keysym::XK_uprightcorner, + //VirtualKeyCode::Upleftcorner => x11::keysym::XK_upleftcorner, + //VirtualKeyCode::Lowleftcorner => x11::keysym::XK_lowleftcorner, + //VirtualKeyCode::Crossinglines => x11::keysym::XK_crossinglines, + //VirtualKeyCode::Horizlinescan1 => x11::keysym::XK_horizlinescan1, + //VirtualKeyCode::Horizlinescan3 => x11::keysym::XK_horizlinescan3, + //VirtualKeyCode::Horizlinescan5 => x11::keysym::XK_horizlinescan5, + //VirtualKeyCode::Horizlinescan7 => x11::keysym::XK_horizlinescan7, + //VirtualKeyCode::Horizlinescan9 => x11::keysym::XK_horizlinescan9, + //VirtualKeyCode::Leftt => x11::keysym::XK_leftt, + //VirtualKeyCode::Rightt => x11::keysym::XK_rightt, + //VirtualKeyCode::Bott => x11::keysym::XK_bott, + //VirtualKeyCode::Topt => x11::keysym::XK_topt, + //VirtualKeyCode::Vertbar => x11::keysym::XK_vertbar, + //VirtualKeyCode::Emspace => x11::keysym::XK_emspace, + //VirtualKeyCode::Enspace => x11::keysym::XK_enspace, + //VirtualKeyCode::Em3space => x11::keysym::XK_em3space, + //VirtualKeyCode::Em4space => x11::keysym::XK_em4space, + //VirtualKeyCode::Digitspace => x11::keysym::XK_digitspace, + //VirtualKeyCode::Punctspace => x11::keysym::XK_punctspace, + //VirtualKeyCode::Thinspace => x11::keysym::XK_thinspace, + //VirtualKeyCode::Hairspace => x11::keysym::XK_hairspace, + //VirtualKeyCode::Emdash => x11::keysym::XK_emdash, + //VirtualKeyCode::Endash => x11::keysym::XK_endash, + //VirtualKeyCode::Signifblank => x11::keysym::XK_signifblank, + //VirtualKeyCode::Ellipsis => x11::keysym::XK_ellipsis, + //VirtualKeyCode::Doubbaselinedot => x11::keysym::XK_doubbaselinedot, + //VirtualKeyCode::Onethird => x11::keysym::XK_onethird, + //VirtualKeyCode::Twothirds => x11::keysym::XK_twothirds, + //VirtualKeyCode::Onefifth => x11::keysym::XK_onefifth, + //VirtualKeyCode::Twofifths => x11::keysym::XK_twofifths, + //VirtualKeyCode::Threefifths => x11::keysym::XK_threefifths, + //VirtualKeyCode::Fourfifths => x11::keysym::XK_fourfifths, + //VirtualKeyCode::Onesixth => x11::keysym::XK_onesixth, + //VirtualKeyCode::Fivesixths => x11::keysym::XK_fivesixths, + //VirtualKeyCode::Careof => x11::keysym::XK_careof, + //VirtualKeyCode::Figdash => x11::keysym::XK_figdash, + //VirtualKeyCode::Leftanglebracket => x11::keysym::XK_leftanglebracket, + //VirtualKeyCode::Decimalpoint => x11::keysym::XK_decimalpoint, + //VirtualKeyCode::Rightanglebracket => x11::keysym::XK_rightanglebracket, + //VirtualKeyCode::Marker => x11::keysym::XK_marker, + //VirtualKeyCode::Oneeighth => x11::keysym::XK_oneeighth, + //VirtualKeyCode::Threeeighths => x11::keysym::XK_threeeighths, + //VirtualKeyCode::Fiveeighths => x11::keysym::XK_fiveeighths, + //VirtualKeyCode::Seveneighths => x11::keysym::XK_seveneighths, + //VirtualKeyCode::Trademark => x11::keysym::XK_trademark, + //VirtualKeyCode::Signaturemark => x11::keysym::XK_signaturemark, + //VirtualKeyCode::Trademarkincircle => x11::keysym::XK_trademarkincircle, + //VirtualKeyCode::Leftopentriangle => x11::keysym::XK_leftopentriangle, + //VirtualKeyCode::Rightopentriangle => x11::keysym::XK_rightopentriangle, + //VirtualKeyCode::Emopencircle => x11::keysym::XK_emopencircle, + //VirtualKeyCode::Emopenrectangle => x11::keysym::XK_emopenrectangle, + //VirtualKeyCode::Leftsinglequotemark => x11::keysym::XK_leftsinglequotemark, + //VirtualKeyCode::Rightsinglequotemark => x11::keysym::XK_rightsinglequotemark, + //VirtualKeyCode::Leftdoublequotemark => x11::keysym::XK_leftdoublequotemark, + //VirtualKeyCode::Rightdoublequotemark => x11::keysym::XK_rightdoublequotemark, + //VirtualKeyCode::Prescription => x11::keysym::XK_prescription, + //VirtualKeyCode::Minutes => x11::keysym::XK_minutes, + //VirtualKeyCode::Seconds => x11::keysym::XK_seconds, + //VirtualKeyCode::Latincross => x11::keysym::XK_latincross, + //VirtualKeyCode::Hexagram => x11::keysym::XK_hexagram, + //VirtualKeyCode::Filledrectbullet => x11::keysym::XK_filledrectbullet, + //VirtualKeyCode::Filledlefttribullet => x11::keysym::XK_filledlefttribullet, + //VirtualKeyCode::Filledrighttribullet => x11::keysym::XK_filledrighttribullet, + //VirtualKeyCode::Emfilledcircle => x11::keysym::XK_emfilledcircle, + //VirtualKeyCode::Emfilledrect => x11::keysym::XK_emfilledrect, + //VirtualKeyCode::Enopencircbullet => x11::keysym::XK_enopencircbullet, + //VirtualKeyCode::Enopensquarebullet => x11::keysym::XK_enopensquarebullet, + //VirtualKeyCode::Openrectbullet => x11::keysym::XK_openrectbullet, + //VirtualKeyCode::Opentribulletup => x11::keysym::XK_opentribulletup, + //VirtualKeyCode::Opentribulletdown => x11::keysym::XK_opentribulletdown, + //VirtualKeyCode::Openstar => x11::keysym::XK_openstar, + //VirtualKeyCode::Enfilledcircbullet => x11::keysym::XK_enfilledcircbullet, + //VirtualKeyCode::Enfilledsqbullet => x11::keysym::XK_enfilledsqbullet, + //VirtualKeyCode::Filledtribulletup => x11::keysym::XK_filledtribulletup, + //VirtualKeyCode::Filledtribulletdown => x11::keysym::XK_filledtribulletdown, + //VirtualKeyCode::Leftpointer => x11::keysym::XK_leftpointer, + //VirtualKeyCode::Rightpointer => x11::keysym::XK_rightpointer, + //VirtualKeyCode::Club => x11::keysym::XK_club, + //VirtualKeyCode::Diamond => x11::keysym::XK_diamond, + //VirtualKeyCode::Heart => x11::keysym::XK_heart, + //VirtualKeyCode::Maltesecross => x11::keysym::XK_maltesecross, + //VirtualKeyCode::Dagger => x11::keysym::XK_dagger, + //VirtualKeyCode::Doubledagger => x11::keysym::XK_doubledagger, + //VirtualKeyCode::Checkmark => x11::keysym::XK_checkmark, + //VirtualKeyCode::Ballotcross => x11::keysym::XK_ballotcross, + //VirtualKeyCode::Musicalsharp => x11::keysym::XK_musicalsharp, + //VirtualKeyCode::Musicalflat => x11::keysym::XK_musicalflat, + //VirtualKeyCode::Malesymbol => x11::keysym::XK_malesymbol, + //VirtualKeyCode::Femalesymbol => x11::keysym::XK_femalesymbol, + //VirtualKeyCode::Telephone => x11::keysym::XK_telephone, + //VirtualKeyCode::Telephonerecorder => x11::keysym::XK_telephonerecorder, + //VirtualKeyCode::Phonographcopyright => x11::keysym::XK_phonographcopyright, + //VirtualKeyCode::Caret => x11::keysym::XK_caret, + //VirtualKeyCode::Singlelowquotemark => x11::keysym::XK_singlelowquotemark, + //VirtualKeyCode::Doublelowquotemark => x11::keysym::XK_doublelowquotemark, + //VirtualKeyCode::Cursor => x11::keysym::XK_cursor, + //VirtualKeyCode::Leftcaret => x11::keysym::XK_leftcaret, + //VirtualKeyCode::Rightcaret => x11::keysym::XK_rightcaret, + //VirtualKeyCode::Downcaret => x11::keysym::XK_downcaret, + //VirtualKeyCode::Upcaret => x11::keysym::XK_upcaret, + //VirtualKeyCode::Overbar => x11::keysym::XK_overbar, + //VirtualKeyCode::Downtack => x11::keysym::XK_downtack, + //VirtualKeyCode::Upshoe => x11::keysym::XK_upshoe, + //VirtualKeyCode::Downstile => x11::keysym::XK_downstile, + //VirtualKeyCode::Underbar => x11::keysym::XK_underbar, + //VirtualKeyCode::Jot => x11::keysym::XK_jot, + //VirtualKeyCode::Quad => x11::keysym::XK_quad, + //VirtualKeyCode::Uptack => x11::keysym::XK_uptack, + //VirtualKeyCode::Circle => x11::keysym::XK_circle, + //VirtualKeyCode::Upstile => x11::keysym::XK_upstile, + //VirtualKeyCode::Downshoe => x11::keysym::XK_downshoe, + //VirtualKeyCode::Rightshoe => x11::keysym::XK_rightshoe, + //VirtualKeyCode::Leftshoe => x11::keysym::XK_leftshoe, + //VirtualKeyCode::Lefttack => x11::keysym::XK_lefttack, + //VirtualKeyCode::Righttack => x11::keysym::XK_righttack, + //VirtualKeyCode::Hebrew_doublelowline => x11::keysym::XK_hebrew_doublelowline, + //VirtualKeyCode::Hebrew_aleph => x11::keysym::XK_hebrew_aleph, + //VirtualKeyCode::Hebrew_bet => x11::keysym::XK_hebrew_bet, + //VirtualKeyCode::Hebrew_beth => x11::keysym::XK_hebrew_beth, + //VirtualKeyCode::Hebrew_gimel => x11::keysym::XK_hebrew_gimel, + //VirtualKeyCode::Hebrew_gimmel => x11::keysym::XK_hebrew_gimmel, + //VirtualKeyCode::Hebrew_dalet => x11::keysym::XK_hebrew_dalet, + //VirtualKeyCode::Hebrew_daleth => x11::keysym::XK_hebrew_daleth, + //VirtualKeyCode::Hebrew_he => x11::keysym::XK_hebrew_he, + //VirtualKeyCode::Hebrew_waw => x11::keysym::XK_hebrew_waw, + //VirtualKeyCode::Hebrew_zain => x11::keysym::XK_hebrew_zain, + //VirtualKeyCode::Hebrew_zayin => x11::keysym::XK_hebrew_zayin, + //VirtualKeyCode::Hebrew_chet => x11::keysym::XK_hebrew_chet, + //VirtualKeyCode::Hebrew_het => x11::keysym::XK_hebrew_het, + //VirtualKeyCode::Hebrew_tet => x11::keysym::XK_hebrew_tet, + //VirtualKeyCode::Hebrew_teth => x11::keysym::XK_hebrew_teth, + //VirtualKeyCode::Hebrew_yod => x11::keysym::XK_hebrew_yod, + //VirtualKeyCode::Hebrew_finalkaph => x11::keysym::XK_hebrew_finalkaph, + //VirtualKeyCode::Hebrew_kaph => x11::keysym::XK_hebrew_kaph, + //VirtualKeyCode::Hebrew_lamed => x11::keysym::XK_hebrew_lamed, + //VirtualKeyCode::Hebrew_finalmem => x11::keysym::XK_hebrew_finalmem, + //VirtualKeyCode::Hebrew_mem => x11::keysym::XK_hebrew_mem, + //VirtualKeyCode::Hebrew_finalnun => x11::keysym::XK_hebrew_finalnun, + //VirtualKeyCode::Hebrew_nun => x11::keysym::XK_hebrew_nun, + //VirtualKeyCode::Hebrew_samech => x11::keysym::XK_hebrew_samech, + //VirtualKeyCode::Hebrew_samekh => x11::keysym::XK_hebrew_samekh, + //VirtualKeyCode::Hebrew_ayin => x11::keysym::XK_hebrew_ayin, + //VirtualKeyCode::Hebrew_finalpe => x11::keysym::XK_hebrew_finalpe, + //VirtualKeyCode::Hebrew_pe => x11::keysym::XK_hebrew_pe, + //VirtualKeyCode::Hebrew_finalzade => x11::keysym::XK_hebrew_finalzade, + //VirtualKeyCode::Hebrew_finalzadi => x11::keysym::XK_hebrew_finalzadi, + //VirtualKeyCode::Hebrew_zade => x11::keysym::XK_hebrew_zade, + //VirtualKeyCode::Hebrew_zadi => x11::keysym::XK_hebrew_zadi, + //VirtualKeyCode::Hebrew_qoph => x11::keysym::XK_hebrew_qoph, + //VirtualKeyCode::Hebrew_kuf => x11::keysym::XK_hebrew_kuf, + //VirtualKeyCode::Hebrew_resh => x11::keysym::XK_hebrew_resh, + //VirtualKeyCode::Hebrew_shin => x11::keysym::XK_hebrew_shin, + //VirtualKeyCode::Hebrew_taw => x11::keysym::XK_hebrew_taw, + //VirtualKeyCode::Hebrew_taf => x11::keysym::XK_hebrew_taf, + //VirtualKeyCode::Hebrew_switch => x11::keysym::XK_Hebrew_switch, + VirtualKeyCode::NavigateBackward => x11::keysym::XF86XK_Back, + VirtualKeyCode::NavigateForward => x11::keysym::XF86XK_Forward, + VirtualKeyCode::Copy => x11::keysym::XF86XK_Copy, + VirtualKeyCode::Paste => x11::keysym::XF86XK_Paste, + VirtualKeyCode::Cut => x11::keysym::XF86XK_Cut, + _ => return None, + }) +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_keysym_to_vkc() { + let keysym: XKeySym = VirtualKeyCode::W.into(); + let keycode: VirtualKeyCode = keysym.into(); + + assert_eq!(keycode, VirtualKeyCode::W); + + let keysym2: XKeySym = keycode.into(); + + assert_eq!(keysym2, keysym); + assert_eq!(&x11::keysym::XK_W, keysym.as_ref()); + } +} diff --git a/src/backends/xlib/mod.rs b/src/backends/xlib/mod.rs new file mode 100644 index 0000000..3d8d78f --- /dev/null +++ b/src/backends/xlib/mod.rs @@ -0,0 +1,978 @@ +use log::{error, warn}; +use std::{ffi::CString, rc::Rc}; + +use thiserror::Error; + +use x11::xlib::{self, Atom, Window, XEvent, XInternAtom, XKeyEvent}; + +use crate::backends::{ + keycodes::KeyOrButton, xlib::keysym::mouse_button_to_xbutton, +}; + +use self::keysym::{ + keysym_to_virtual_keycode, virtual_keycode_to_keysym, xev_to_mouse_button, + XKeySym, +}; + +use super::{ + keycodes::VirtualKeyCode, + window_event::{ + ButtonEvent, ConfigureEvent, DestroyEvent, EnterEvent, KeyEvent, + KeyOrMouseBind, KeyState, MapEvent, ModifierState, MotionEvent, Point, + UnmapEvent, WindowEvent, + }, + WindowServerBackend, +}; + +pub mod color; +pub mod keysym; + +pub type XLibWindowEvent = WindowEvent; + +#[derive(Clone)] +pub struct Display(Rc<*mut x11::xlib::Display>); + +#[derive(Debug, Error)] +pub enum XlibError { + #[error("BadAccess")] + BadAccess, + #[error("BadAlloc")] + BadAlloc, + #[error("BadAtom")] + BadAtom, + #[error("BadColor")] + BadColor, + #[error("BadCursor")] + BadCursor, + #[error("BadDrawable")] + BadDrawable, + #[error("BadFont")] + BadFont, + #[error("BadGC")] + BadGC, + #[error("BadIDChoice")] + BadIDChoice, + #[error("BadImplementation")] + BadImplementation, + #[error("BadLength")] + BadLength, + #[error("BadMatch")] + BadMatch, + #[error("BadName")] + BadName, + #[error("BadPixmap")] + BadPixmap, + #[error("BadRequest")] + BadRequest, + #[error("BadValue")] + BadValue, + #[error("BadWindow")] + BadWindow, + #[error("Invalid XError: {0}")] + InvalidError(u8), +} + +impl From for XlibError { + fn from(value: u8) -> Self { + match value { + xlib::BadAccess => XlibError::BadAccess, + xlib::BadAlloc => XlibError::BadAlloc, + xlib::BadAtom => XlibError::BadAtom, + xlib::BadColor => XlibError::BadColor, + xlib::BadCursor => XlibError::BadCursor, + xlib::BadDrawable => XlibError::BadDrawable, + xlib::BadFont => XlibError::BadFont, + xlib::BadGC => XlibError::BadGC, + xlib::BadIDChoice => XlibError::BadIDChoice, + xlib::BadImplementation => XlibError::BadImplementation, + xlib::BadLength => XlibError::BadLength, + xlib::BadMatch => XlibError::BadMatch, + xlib::BadName => XlibError::BadName, + xlib::BadPixmap => XlibError::BadPixmap, + xlib::BadRequest => XlibError::BadRequest, + xlib::BadValue => XlibError::BadValue, + xlib::BadWindow => XlibError::BadWindow, + any => XlibError::InvalidError(any), + } + } +} + +impl Display { + pub fn new(display: *mut x11::xlib::Display) -> Self { + Self { + 0: Rc::new(display), + } + } + + pub fn get(&self) -> *mut x11::xlib::Display { + *self.0 + } +} + +pub struct XLib { + display: Display, + root: Window, + screen: i32, + atoms: XLibAtoms, + keybinds: Vec, + active_border_color: Option, + inactive_border_color: Option, +} + +impl Drop for XLib { + fn drop(&mut self) { + self.close_dpy(); + } +} + +impl XLib { + fn new() -> Self { + let (display, screen, root) = { + let display = unsafe { xlib::XOpenDisplay(std::ptr::null()) }; + assert_ne!(display, std::ptr::null_mut()); + let screen = unsafe { xlib::XDefaultScreen(display) }; + let root = unsafe { xlib::XRootWindow(display, screen) }; + + let display = Display::new(display); + + (display, screen, root) + }; + + let atoms = XLibAtoms::init(display.clone()); + + Self { + display, + screen, + root, + atoms, + keybinds: Vec::new(), + active_border_color: None, + inactive_border_color: None, + } + } + + unsafe fn init_as_wm(&self) { + let mut window_attributes = + std::mem::MaybeUninit::::zeroed() + .assume_init(); + + window_attributes.event_mask = xlib::SubstructureRedirectMask + | xlib::StructureNotifyMask + | xlib::SubstructureNotifyMask + | xlib::EnterWindowMask + | xlib::PointerMotionMask + | xlib::ButtonPressMask; + + xlib::XChangeWindowAttributes( + self.dpy(), + self.root, + xlib::CWEventMask, + &mut window_attributes, + ); + + xlib::XSelectInput(self.dpy(), self.root, window_attributes.event_mask); + xlib::XSetErrorHandler(Some(xlib_error_handler)); + xlib::XSync(self.dpy(), 0); + } + + fn dpy(&self) -> *mut xlib::Display { + self.display.get() + } + + fn close_dpy(&self) { + unsafe { + xlib::XCloseDisplay(self.dpy()); + } + } + + fn next_xevent(&mut self) -> XEvent { + let event = unsafe { + let mut event = std::mem::MaybeUninit::::zeroed(); + xlib::XNextEvent(self.dpy(), event.as_mut_ptr()); + + event.assume_init() + }; + + // match event.get_type() { + // xlib::KeyPress | xlib::KeyRelease => { + // self.update_modifier_state(AsRef::::as_ref( + // &event, + // )); + // } + // _ => {} + // } + + event + } + + fn xevent_to_window_event(&self, event: XEvent) -> Option { + match event.get_type() { + xlib::MapRequest => { + let ev = unsafe { &event.map_request }; + Some(XLibWindowEvent::MapRequestEvent(MapEvent { + window: ev.window, + })) + } + xlib::UnmapNotify => { + let ev = unsafe { &event.unmap }; + Some(XLibWindowEvent::UnmapEvent(UnmapEvent { + window: ev.window, + })) + } + xlib::ConfigureRequest => { + let ev = unsafe { &event.configure_request }; + Some(XLibWindowEvent::ConfigureEvent(ConfigureEvent { + window: ev.window, + position: (ev.x, ev.y).into(), + size: (ev.width, ev.height).into(), + })) + } + xlib::EnterNotify => { + let ev = unsafe { &event.crossing }; + Some(XLibWindowEvent::EnterEvent(EnterEvent { + window: ev.window, + })) + } + xlib::DestroyNotify => { + let ev = unsafe { &event.destroy_window }; + Some(XLibWindowEvent::DestroyEvent(DestroyEvent { + window: ev.window, + })) + } + xlib::MotionNotify => { + let ev = unsafe { &event.motion }; + Some(XLibWindowEvent::MotionEvent(MotionEvent { + position: (ev.x, ev.y).into(), + window: ev.window, + })) + } + // both ButtonPress and ButtonRelease use the XButtonEvent structure, aliased as either + // XButtonReleasedEvent or XButtonPressedEvent + xlib::ButtonPress | xlib::ButtonRelease => { + let ev = unsafe { &event.button }; + let keycode = xev_to_mouse_button(ev).unwrap(); + let state = if ev.type_ == xlib::ButtonPress { + KeyState::Pressed + } else { + KeyState::Released + }; + + Some(XLibWindowEvent::ButtonEvent(ButtonEvent::new( + ev.subwindow, + state, + keycode, + (ev.x, ev.y).into(), + ModifierState::from_modmask(ev.state), + ))) + } + xlib::KeyPress | xlib::KeyRelease => { + let ev = unsafe { &event.key }; + + let keycode = + keysym_to_virtual_keycode(self.keyev_to_keysym(ev).get()); + let state = if ev.type_ == xlib::KeyPress { + KeyState::Pressed + } else { + KeyState::Released + }; + + keycode.map(|keycode| { + XLibWindowEvent::KeyEvent(KeyEvent::new( + ev.subwindow, + state, + keycode, + ModifierState::from_modmask(ev.state), + )) + }) + } + _ => None, + } + } + + #[allow(dead_code)] + fn get_window_attributes( + &self, + window: xlib::Window, + ) -> Option { + let mut wa = unsafe { + std::mem::MaybeUninit::::zeroed() + .assume_init() + }; + + if unsafe { + xlib::XGetWindowAttributes(self.dpy(), window, &mut wa) != 0 + } { + Some(wa) + } else { + None + } + } + + fn check_for_protocol( + &self, + window: xlib::Window, + proto: xlib::Atom, + ) -> bool { + let mut protos: *mut xlib::Atom = std::ptr::null_mut(); + let mut num_protos: i32 = 0; + + unsafe { + if xlib::XGetWMProtocols( + self.dpy(), + window, + &mut protos, + &mut num_protos, + ) != 0 + { + for i in 0..num_protos { + if *protos.offset(i as isize) == proto { + return true; + } + } + } + } + + return false; + } + + fn send_protocol(&self, window: xlib::Window, proto: Atom) -> bool { + if self.check_for_protocol(window, proto) { + let mut data = xlib::ClientMessageData::default(); + data.set_long(0, proto as i64); + + let mut event = XEvent { + client_message: xlib::XClientMessageEvent { + type_: xlib::ClientMessage, + serial: 0, + display: self.dpy(), + send_event: 0, + window, + format: 32, + message_type: self.atoms.wm_protocols, + data, + }, + }; + + unsafe { + xlib::XSendEvent( + self.dpy(), + window, + 0, + xlib::NoEventMask, + &mut event, + ); + } + + true + } else { + false + } + } + + // #[allow(non_upper_case_globals)] + // fn update_modifier_state(&mut self, keyevent: &XKeyEvent) { + // //keyevent.keycode + // let keysym = self.keyev_to_keysym(keyevent); + + // use x11::keysym::*; + + // let modifier = match keysym.get() { + // XK_Shift_L | XK_Shift_R => Some(ModifierKey::Shift), + // XK_Control_L | XK_Control_R => Some(ModifierKey::Control), + // XK_Alt_L | XK_Alt_R => Some(ModifierKey::Alt), + // XK_ISO_Level3_Shift => Some(ModifierKey::AltGr), + // XK_Caps_Lock => Some(ModifierKey::ShiftLock), + // XK_Num_Lock => Some(ModifierKey::NumLock), + // XK_Win_L | XK_Win_R => Some(ModifierKey::Super), + // XK_Super_L | XK_Super_R => Some(ModifierKey::Super), + // _ => None, + // }; + + // if let Some(modifier) = modifier { + // match keyevent.type_ { + // KeyPress => self.modifier_state.insert_mod(modifier), + // KeyRelease => self.modifier_state.unset_mod(modifier), + // _ => unreachable!("keyyevent != (KeyPress | KeyRelease)"), + // } + // } + // } + + fn get_numlock_mask(&self) -> Option { + unsafe { + let modmap = xlib::XGetModifierMapping(self.dpy()); + let max_keypermod = (*modmap).max_keypermod; + + for i in 0..8 { + for j in 0..max_keypermod { + if *(*modmap) + .modifiermap + .offset((i * max_keypermod + j) as isize) + == xlib::XKeysymToKeycode( + self.dpy(), + x11::keysym::XK_Num_Lock as u64, + ) + { + return Some(1 << i); + } + } + } + } + + None + } + + fn grab_key_or_button( + &self, + binding: &KeyOrMouseBind, + window: xlib::Window, + ) { + let modmask = binding.modifiers.as_modmask(self); + + let numlock_mask = self + .get_numlock_mask() + .expect("failed to query numlock mask."); + + let modifiers = vec![ + 0, + xlib::LockMask, + numlock_mask, + xlib::LockMask | numlock_mask, + ]; + + let keycode = match binding.key { + KeyOrButton::Key(key) => self.vk_to_keycode(key), + KeyOrButton::Button(button) => mouse_button_to_xbutton(button), + }; + + for modifier in modifiers.iter() { + match binding.key { + KeyOrButton::Key(_) => unsafe { + xlib::XGrabKey( + self.dpy(), + keycode, + modmask | modifier, + window, + 1, + xlib::GrabModeAsync, + xlib::GrabModeAsync, + ); + }, + KeyOrButton::Button(_) => unsafe { + xlib::XGrabButton( + self.dpy(), + keycode as u32, + modmask | modifier, + window, + 1, + (xlib::ButtonPressMask + | xlib::ButtonReleaseMask + | xlib::PointerMotionMask) + as u32, + xlib::GrabModeAsync, + xlib::GrabModeAsync, + 0, + 0, + ); + }, + } + } + } + + #[allow(dead_code)] + fn ungrab_key_or_button( + &self, + binding: &KeyOrMouseBind, + window: xlib::Window, + ) { + let modmask = binding.modifiers.as_modmask(self); + + let numlock_mask = self + .get_numlock_mask() + .expect("failed to query numlock mask."); + + let modifiers = vec![ + 0, + xlib::LockMask, + numlock_mask, + xlib::LockMask | numlock_mask, + ]; + + let keycode = match binding.key { + KeyOrButton::Key(key) => self.vk_to_keycode(key), + KeyOrButton::Button(button) => mouse_button_to_xbutton(button), + }; + + for modifier in modifiers.iter() { + match binding.key { + KeyOrButton::Key(_) => unsafe { + xlib::XUngrabKey( + self.dpy(), + keycode, + modmask | modifier, + window, + ); + }, + KeyOrButton::Button(_) => unsafe { + xlib::XUngrabButton( + self.dpy(), + keycode as u32, + modmask | modifier, + window, + ); + }, + } + } + } + + fn grab_global_keybinds(&self, window: xlib::Window) { + for binding in self.keybinds.iter() { + self.grab_key_or_button(binding, window); + } + } + + fn vk_to_keycode(&self, vk: VirtualKeyCode) -> i32 { + unsafe { + xlib::XKeysymToKeycode( + self.dpy(), + virtual_keycode_to_keysym(vk).unwrap() as u64, + ) as i32 + } + } + + fn keyev_to_keysym(&self, ev: &XKeyEvent) -> XKeySym { + let keysym = + unsafe { xlib::XLookupKeysym(ev as *const _ as *mut _, 0) }; + + XKeySym::new(keysym as u32) + } +} + +trait ModifierStateExt { + fn as_modmask(&self, xlib: &XLib) -> u32; + fn from_modmask(modmask: u32) -> Self; +} + +impl ModifierStateExt for ModifierState { + fn as_modmask(&self, xlib: &XLib) -> u32 { + let mut mask = 0; + let _numlock_mask = xlib + .get_numlock_mask() + .expect("failed to query numlock mask"); + + mask |= xlib::ShiftMask * u32::from(self.contains(Self::SHIFT)); + //mask |= xlib::LockMask * u32::from(self.contains(Self::SHIFT_LOCK)); + mask |= xlib::ControlMask * u32::from(self.contains(Self::CONTROL)); + mask |= xlib::Mod1Mask * u32::from(self.contains(Self::ALT)); + //mask |= xlib::Mod2Mask * u32::from(self.contains(Self::NUM_LOCK)); + //mask |= xlib::Mod3Mask * u32::from(self.contains(Self::ALT_GR)); + mask |= xlib::Mod4Mask * u32::from(self.contains(Self::SUPER)); + //mask |= numlock_mask * u32::from(self.contains(Self::NUM_LOCK)); + + mask + } + + fn from_modmask(modmask: u32) -> Self { + let mut state = Self::empty(); + state.set(Self::SHIFT, (modmask & xlib::ShiftMask) != 0); + //state.set(Self::SHIFT_LOCK, (modmask & xlib::LockMask) != 0); + state.set(Self::CONTROL, (modmask & xlib::ControlMask) != 0); + state.set(Self::ALT, (modmask & xlib::Mod1Mask) != 0); + //state.set(Self::NUM_LOCK, (modmask & xlib::Mod2Mask) != 0); + state.set(Self::ALT_GR, (modmask & xlib::Mod3Mask) != 0); + state.set(Self::SUPER, (modmask & xlib::Mod4Mask) != 0); + + state + } +} + +impl WindowServerBackend for XLib { + type Window = xlib::Window; + + fn build() -> Self { + let xlib = Self::new(); + unsafe { xlib.init_as_wm() }; + xlib + } + + fn next_event(&mut self) -> super::window_event::WindowEvent { + loop { + let ev = self.next_xevent(); + let ev = self.xevent_to_window_event(ev); + + if let Some(ev) = ev { + return ev; + } + } + } + + fn handle_event( + &mut self, + event: super::window_event::WindowEvent, + ) { + match event { + WindowEvent::MapRequestEvent(event) => { + unsafe { + xlib::XMapWindow(self.dpy(), event.window); + + xlib::XSelectInput( + self.dpy(), + event.window, + xlib::EnterWindowMask + | xlib::FocusChangeMask + | xlib::PropertyChangeMask + | xlib::StructureNotifyMask, + ); + } + + self.grab_global_keybinds(event.window); + } + WindowEvent::ConfigureEvent(event) => { + self.configure_window( + event.window, + Some(event.size), + Some(event.position), + None, + ); + } + _ => {} + } + } + + fn add_keybind(&mut self, keybind: super::window_event::KeyOrMouseBind) { + self.grab_key_or_button(&keybind, self.root); + self.keybinds.push(keybind); + } + + fn remove_keybind( + &mut self, + keybind: &super::window_event::KeyOrMouseBind, + ) { + self.keybinds.retain(|kb| kb != keybind); + } + + fn focus_window(&self, window: Self::Window) { + unsafe { + xlib::XSetInputFocus( + self.dpy(), + window, + xlib::RevertToPointerRoot, + xlib::CurrentTime, + ); + + let border_color = self + .active_border_color + .as_ref() + .map(|color| color.pixel()) + .unwrap_or_else(|| { + xlib::XDefaultScreenOfDisplay(self.dpy()) + .as_ref() + .unwrap() + .white_pixel + }); + + xlib::XSetWindowBorder(self.dpy(), window, border_color); + + xlib::XChangeProperty( + self.dpy(), + self.root, + self.atoms.wm_active_window, + xlib::XA_WINDOW, + 32, + xlib::PropModeReplace, + &window as *const u64 as *const _, + 1, + ); + } + + self.send_protocol(window, self.atoms.wm_take_focus); + } + + fn unfocus_window(&self, window: Self::Window) { + unsafe { + xlib::XSetInputFocus( + self.dpy(), + self.root, + xlib::RevertToPointerRoot, + xlib::CurrentTime, + ); + + // TODO: make painting the window border a seperate function, and configurable + + let border_color = self + .inactive_border_color + .as_ref() + .map(|color| color.pixel()) + .unwrap_or_else(|| { + xlib::XDefaultScreenOfDisplay(self.dpy()) + .as_ref() + .unwrap() + .black_pixel + }); + + xlib::XSetWindowBorder(self.dpy(), window, border_color); + + xlib::XDeleteProperty( + self.dpy(), + self.root, + self.atoms.wm_active_window, + ); + } + } + + fn raise_window(&self, window: Self::Window) { + unsafe { + xlib::XRaiseWindow(self.dpy(), window); + } + } + + fn hide_window(&self, window: Self::Window) { + let screen_size = self.screen_size(); + self.move_window(window, screen_size); + } + + fn kill_window(&self, window: Self::Window) { + if !self.send_protocol(window, self.atoms.wm_delete_window) { + unsafe { + xlib::XKillClient(self.dpy(), window); + } + } + } + + fn get_parent_window(&self, window: Self::Window) -> Option { + let mut parent_window: Self::Window = 0; + if unsafe { + xlib::XGetTransientForHint(self.dpy(), window, &mut parent_window) + != 0 + } { + Some(parent_window) + } else { + None + } + } + + fn configure_window( + &self, + window: Self::Window, + new_size: Option>, + new_pos: Option>, + new_border: Option, + ) { + let position = new_pos.unwrap_or(Point::new(0, 0)); + let size = new_size.unwrap_or(Point::new(0, 0)); + let mut wc = xlib::XWindowChanges { + x: position.x, + y: position.y, + width: size.x, + height: size.y, + border_width: new_border.unwrap_or(0), + sibling: 0, + stack_mode: 0, + }; + + let mask = { + let mut mask = 0; + if new_pos.is_some() { + mask |= xlib::CWX | xlib::CWY; + } + if new_size.is_some() { + mask |= xlib::CWWidth | xlib::CWHeight; + } + if new_border.is_some() { + mask |= xlib::CWBorderWidth; + } + + u32::from(mask) + }; + + unsafe { + xlib::XConfigureWindow(self.dpy(), window, mask, &mut wc); + } + } + + fn screen_size(&self) -> Point { + unsafe { + let mut wa = + std::mem::MaybeUninit::::zeroed(); + + xlib::XGetWindowAttributes(self.dpy(), self.root, wa.as_mut_ptr()); + + let wa = wa.assume_init(); + + (wa.width, wa.height).into() + } + } + + fn get_window_size(&self, window: Self::Window) -> Option> { + self.get_window_attributes(window) + .map(|wa| (wa.width, wa.height).into()) + } + + fn grab_cursor(&self) { + unsafe { + xlib::XGrabPointer( + self.dpy(), + self.root, + 0, + (xlib::ButtonPressMask + | xlib::ButtonReleaseMask + | xlib::PointerMotionMask) as u32, + xlib::GrabModeAsync, + xlib::GrabModeAsync, + 0, + 0, + xlib::CurrentTime, + ); + } + } + + fn ungrab_cursor(&self) { + unsafe { + xlib::XUngrabPointer(self.dpy(), xlib::CurrentTime); + } + } + + fn move_cursor(&self, window: Option, position: Point) { + unsafe { + xlib::XWarpPointer( + self.dpy(), + 0, + window.unwrap_or(self.root), + 0, + 0, + 0, + 0, + position.x, + position.y, + ); + } + } + + fn all_windows(&self) -> Option> { + let mut parent = 0; + let mut root = 0; + let mut children = std::ptr::null_mut(); + let mut num_children = 0; + + unsafe { + xlib::XQueryTree( + self.dpy(), + self.root, + &mut root, + &mut parent, + &mut children, + &mut num_children, + ) != 0 + } + .then(|| { + let windows = unsafe { + std::slice::from_raw_parts(children, num_children as usize) + .to_vec() + }; + + unsafe { xlib::XFree(children as *mut _) }; + + windows + }) + } + + fn set_active_window_border_color(&mut self, color_name: &str) { + self.active_border_color = color::XftColor::new( + self.display.clone(), + self.screen, + color_name.to_owned(), + ) + .ok(); + } + + fn set_inactive_window_border_color(&mut self, color_name: &str) { + self.inactive_border_color = color::XftColor::new( + self.display.clone(), + self.screen, + color_name.to_owned(), + ) + .ok(); + } +} + +#[allow(dead_code)] +struct XLibAtoms { + wm_protocols: Atom, + wm_delete_window: Atom, + wm_active_window: Atom, + wm_take_focus: Atom, + net_supported: Atom, + net_active_window: Atom, + net_client_list: Atom, + net_wm_name: Atom, + net_wm_state: Atom, + net_wm_state_fullscreen: Atom, + net_wm_window_type: Atom, + net_wm_window_type_dialog: Atom, +} + +impl XLibAtoms { + fn init(display: Display) -> Self { + Self { + wm_protocols: Self::get_atom(&display, "WM_PROTOCOLS").unwrap(), + wm_delete_window: Self::get_atom(&display, "WM_DELETE_WINDOW") + .unwrap(), + wm_active_window: Self::get_atom(&display, "WM_ACTIVE_WINDOW") + .unwrap(), + wm_take_focus: Self::get_atom(&display, "WM_TAKE_FOCUS").unwrap(), + net_supported: Self::get_atom(&display, "_NET_SUPPORTED").unwrap(), + net_active_window: Self::get_atom(&display, "_NET_ACTIVE_WINDOW") + .unwrap(), + net_client_list: Self::get_atom(&display, "_NET_CLIENT_LIST") + .unwrap(), + net_wm_name: Self::get_atom(&display, "_NET_WM_NAME").unwrap(), + net_wm_state: Self::get_atom(&display, "_NET_WM_STATE").unwrap(), + net_wm_state_fullscreen: Self::get_atom( + &display, + "_NET_WM_STATE_FULLSCREEN", + ) + .unwrap(), + net_wm_window_type: Self::get_atom(&display, "_NET_WM_WINDOW_TYPE") + .unwrap(), + net_wm_window_type_dialog: Self::get_atom( + &display, + "_NET_WM_WINDOW_TYPE_DIALOG", + ) + .unwrap(), + } + } + + fn get_atom(display: &Display, atom: &str) -> Option { + let name = CString::new(atom).ok()?; + match unsafe { XInternAtom(display.get(), name.as_c_str().as_ptr(), 0) } + { + 0 => None, + atom => Some(atom), + } + } +} + +#[allow(dead_code)] +unsafe extern "C" fn xlib_error_handler( + _dpy: *mut x11::xlib::Display, + ee: *mut x11::xlib::XErrorEvent, +) -> std::os::raw::c_int { + let err_event = ee.as_ref().unwrap(); + let err = XlibError::from(err_event.error_code); + + match err { + err @ XlibError::BadAccess + | err @ XlibError::BadMatch + | err @ XlibError::BadWindow + | err @ XlibError::BadDrawable => { + warn!("{:?}", err); + 0 + } + _ => { + error!( + "wm: fatal error:\nrequest_code: {}\nerror_code: {}", + err_event.request_code, err_event.error_code + ); + std::process::exit(1) + } + } +} diff --git a/src/clients.rs b/src/clients.rs index cd65dc1..e641d8a 100644 --- a/src/clients.rs +++ b/src/clients.rs @@ -4,6 +4,7 @@ use std::{ops::Rem, usize}; use indexmap::IndexMap; use log::{error, info}; +use crate::backends::window_event::Point; use crate::util::BuildIdentityHasher; mod client { @@ -11,11 +12,13 @@ mod client { use x11::xlib::Window; + use crate::backends::window_event::Point; + #[derive(Clone, Debug)] pub struct Client { pub(crate) window: Window, - pub(crate) size: (i32, i32), - pub(crate) position: (i32, i32), + pub(crate) size: Point, + pub(crate) position: Point, pub(crate) transient_for: Option, } @@ -23,8 +26,8 @@ mod client { fn default() -> Self { Self { window: 0, - size: (100, 100), - position: (0, 0), + size: (100, 100).into(), + position: (0, 0).into(), transient_for: None, } } @@ -34,8 +37,8 @@ mod client { #[allow(dead_code)] pub fn new( window: Window, - size: (i32, i32), - position: (i32, i32), + size: Point, + position: Point, ) -> Self { Self { window, @@ -47,7 +50,7 @@ mod client { pub fn new_transient( window: Window, - size: (i32, i32), + size: Point, transient: Window, ) -> Self { Self { @@ -140,7 +143,7 @@ pub struct ClientState { pub(self) virtual_screens: VirtualScreenStore, pub(self) gap: i32, - pub(self) screen_size: (i32, i32), + pub(self) screen_size: Point, pub(self) master_size: f32, border_size: i32, } @@ -166,7 +169,7 @@ impl Default for ClientState { focused: None, virtual_screens: VirtualScreenStore::new(1), gap: 0, - screen_size: (1, 1), + screen_size: (1, 1).into(), master_size: 1.0, border_size: 0, } @@ -189,7 +192,7 @@ impl ClientState { } } - pub fn with_screen_size(self, screen_size: (i32, i32)) -> Self { + pub fn with_screen_size(self, screen_size: Point) -> Self { Self { screen_size, ..self @@ -207,6 +210,7 @@ impl ClientState { self.border_size } + #[allow(dead_code)] pub fn set_border_mut(&mut self, new: i32) { self.border_size = new; } @@ -221,11 +225,12 @@ impl ClientState { client.position = { ( - transient.position.0 - + (transient.size.0 - client.size.0) / 2, - transient.position.1 - + (transient.size.1 - client.size.1) / 2, + transient.position.x + + (transient.size.x - client.size.x) / 2, + transient.position.y + + (transient.size.y - client.size.y) / 2, ) + .into() }; self.floating_clients.insert(key, client); @@ -283,7 +288,7 @@ impl ClientState { .filter(move |&(k, _)| self.is_client_visible(k)) } - fn iter_all_clients(&self) -> impl Iterator { + pub fn iter_all_clients(&self) -> impl Iterator { self.floating_clients.iter().chain(self.clients.iter()) } @@ -532,7 +537,9 @@ impl ClientState { { let (new, old) = self.focus_client_inner(key); - info!("Swapping focus: new({:?}) old({:?})", new, old); + if !(new.is_vacant() && old.is_vacant()) { + info!("Swapping focus: new({:?}) old({:?})", new, old); + } (new, old) } @@ -622,7 +629,7 @@ impl ClientState { */ pub fn arrange_virtual_screen(&mut self) { let gap = self.gap; - let (width, height) = self.screen_size; + let (width, height) = self.screen_size.as_tuple(); // should be fine to unwrap since we will always have at least 1 virtual screen let vs = self.virtual_screens.get_mut_current(); @@ -669,8 +676,8 @@ impl ClientState { if let Some(client) = self.clients.get_mut(key) { *client = Client { - size, - position, + size: size.into(), + position: position.into(), ..*client }; } @@ -688,8 +695,8 @@ impl ClientState { if let Some(client) = self.clients.get_mut(key) { *client = Client { - size, - position, + size: size.into(), + position: position.into(), ..*client }; } @@ -851,7 +858,7 @@ impl VirtualScreenStore { fn go_to_nth(&mut self, n: usize) -> usize { self.last_idx = Some(self.current_idx); - self.current_idx = n % self.screens.len(); + self.current_idx = n.min(self.screens.len() - 1); self.current_idx } diff --git a/src/main.rs b/src/main.rs index 4825886..71025a7 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,3 +1,5 @@ +use std::io::Read; + use log::{debug, error, info, trace, warn}; use log4rs::{ append::{console::ConsoleAppender, file::FileAppender}, @@ -7,11 +9,30 @@ use log4rs::{ }; use state::WMConfig; +mod backends; mod clients; -//mod clients2; mod state; mod util; -mod xlib; + +pub mod error { + use thiserror::Error; + + #[derive(Debug, Error)] + pub enum Error { + #[error("placeholder error for Result as Option")] + NonError, + #[error("Unknown Event")] + UnknownEvent, + #[error("Unhandled VirtualKeyCode")] + UnhandledVirtualKeyCode, + #[error(transparent)] + IoError(#[from] std::io::Error), + #[error(transparent)] + FmtError(#[from] std::fmt::Error), + #[error(transparent)] + XlibError(#[from] crate::backends::xlib::XlibError), + } +} fn init_logger() { let encoder = Box::new(PatternEncoder::new( @@ -46,7 +67,23 @@ fn main() { log_prologue(); - state::WindowManager::new(WMConfig::default()).run(); + let mut config_path = std::path::PathBuf::from(env!("HOME")); + config_path.push(".config/nowm.toml"); + + let config = std::fs::File::open(config_path) + .and_then(|mut file| { + let mut content = String::new(); + file.read_to_string(&mut content)?; + Ok(content) + }) + .and_then(|content| Ok(toml::from_str::(&content)?)) + .unwrap_or_else(|e| { + warn!("error parsing config file: {}", e); + info!("falling back to default config."); + WMConfig::default() + }); + + state::WindowManager::::new(config).run(); } fn log_prologue() { diff --git a/src/state.rs b/src/state.rs index 04c49ee..325c4eb 100644 --- a/src/state.rs +++ b/src/state.rs @@ -1,38 +1,81 @@ -use std::rc::Rc; +use std::{cell::RefCell, rc::Rc}; use log::{error, info}; -use x11::xlib::{ - self, Mod4Mask, ShiftMask, Window, XButtonPressedEvent, - XButtonReleasedEvent, XEvent, XKeyEvent, XMotionEvent, -}; -use xlib::{ - ButtonPressMask, ButtonReleaseMask, PointerMotionMask, - XConfigureRequestEvent, XCrossingEvent, XDestroyWindowEvent, - XMapRequestEvent, XUnmapEvent, -}; +use x11::xlib::{self, Window}; use crate::{ + backends::{ + keycodes::{MouseButton, VirtualKeyCode}, + window_event::{ + ButtonEvent, ConfigureEvent, KeyBind, KeyEvent, KeyState, MapEvent, + ModifierKey, ModifierState, MotionEvent, MouseBind, Point, + WindowEvent, + }, + xlib::XLib, + WindowServerBackend, + }, clients::{Client, ClientEntry, ClientKey, ClientState}, - xlib::KeyOrButton, - xlib::XLib, }; +use serde::Deserialize; + /** Contains static config data for the window manager, the sort of stuff you might want to be able to configure in a config file. -*/ + */ +#[derive(Debug, Deserialize)] pub struct WMConfig { num_virtualscreens: usize, - mod_key: u32, + mod_key: ModifierKey, gap: Option, + kill_clients_on_exit: bool, + #[serde(default = "WMConfig::default_active_window_border_color")] + active_window_border_color: String, + #[serde(default = "WMConfig::default_inactive_window_border_color")] + inactive_window_border_color: String, + #[serde(default = "WMConfig::default_terminal")] + terminal_command: (String, Vec), } -pub struct WindowManager { +impl WMConfig { + fn default_active_window_border_color() -> String { + "#ffffff".to_string() + } + + fn default_inactive_window_border_color() -> String { + "#444444".to_string() + } + + fn default_terminal() -> (String, Vec) { + ("alacritty".to_string(), vec![]) + } +} + +impl Default for WMConfig { + fn default() -> Self { + Self { + num_virtualscreens: 10, + mod_key: ModifierKey::Super, + gap: Some(2), + kill_clients_on_exit: false, + active_window_border_color: + Self::default_active_window_border_color(), + inactive_window_border_color: + Self::default_inactive_window_border_color(), + terminal_command: Self::default_terminal(), + } + } +} + +pub struct WindowManager +where + B: WindowServerBackend, +{ clients: ClientState, move_resize_window: MoveResizeInfo, - keybinds: Vec, - xlib: XLib, + keybinds: Rc>>>, + backend: B, config: WMConfig, } @@ -51,66 +94,91 @@ enum MoveResizeInfo { None, } +#[derive(Debug)] struct MoveInfoInner { window: Window, - starting_cursor_pos: (i32, i32), - starting_window_pos: (i32, i32), + starting_cursor_pos: Point, + starting_window_pos: Point, } +#[derive(Debug)] struct ResizeInfoInner { window: Window, - starting_cursor_pos: (i32, i32), - starting_window_size: (i32, i32), + starting_cursor_pos: Point, + starting_window_size: Point, } -#[derive(Clone)] -struct KeyBinding { - key: KeyOrButton, - closure: Rc, +use derivative::*; + +#[derive(Derivative)] +#[derivative(Clone(bound = ""))] +struct KeyBinding { + key: KeyBind, + closure: Rc, &KeyEvent)>, } -impl WindowManager { +impl KeyBinding { + pub fn new(key: KeyBind, cb: F) -> Self + where + F: Fn(&mut WindowManager, &KeyEvent), + F: 'static, + { + Self { + key, + closure: Rc::new(cb), + } + } + + pub fn call(&self, wm: &mut WindowManager, ev: &KeyEvent) { + (self.closure)(wm, ev); + } +} + +impl WindowManager +where + B: WindowServerBackend, +{ pub fn new(config: WMConfig) -> Self { - let xlib = XLib::new(); + let backend = B::build(); let clients = ClientState::new() .with_virtualscreens(config.num_virtualscreens) .with_gap(config.gap.unwrap_or(1)) .with_border(1) - .with_screen_size(xlib.dimensions()); + .with_screen_size(backend.screen_size()); Self { clients, move_resize_window: MoveResizeInfo::None, - keybinds: Vec::new(), - xlib, + keybinds: Rc::new(RefCell::new(Vec::new())), + backend, config, } .init() } fn init(mut self) -> Self { - self.xlib.add_global_keybind(KeyOrButton::button( - 1, - self.config.mod_key, - ButtonPressMask | ButtonReleaseMask | PointerMotionMask, - )); - self.xlib.add_global_keybind(KeyOrButton::button( - 2, - self.config.mod_key, - ButtonPressMask | ButtonReleaseMask | PointerMotionMask, - )); - self.xlib.add_global_keybind(KeyOrButton::button( - 3, - self.config.mod_key, - ButtonPressMask | ButtonReleaseMask | PointerMotionMask, - )); + self.backend.add_keybind( + MouseBind::new(MouseButton::Left) + .with_mod(self.config.mod_key) + .into(), + ); + self.backend.add_keybind( + MouseBind::new(MouseButton::Middle) + .with_mod(self.config.mod_key) + .into(), + ); + self.backend.add_keybind( + MouseBind::new(MouseButton::Right) + .with_mod(self.config.mod_key) + .into(), + ); self.add_keybind(KeyBinding::new( - self.xlib.make_key("P", self.config.mod_key), + KeyBind::new(VirtualKeyCode::P).with_mod(self.config.mod_key), |wm, _| { wm.spawn( - "dmenu_run", + &"dmenu_run", &[ "-m", "0", @@ -129,23 +197,23 @@ impl WindowManager { }, )); + // self.add_keybind(KeyBinding::new( + // KeyBind::new(VirtualKeyCode::Print), + // |wm, _| wm.spawn("screenshot.sh", &[]), + // )); + + // self.add_keybind(KeyBinding::new( + // KeyBind::new(VirtualKeyCode::Print).with_mod(ModifierKey::Shift), + // |wm, _| wm.spawn("screenshot.sh", &["-edit"]), + // )); + self.add_keybind(KeyBinding::new( - self.xlib.make_key("Print", 0), - |wm, _| wm.spawn("screenshot.sh", &[]), + KeyBind::new(VirtualKeyCode::M).with_mod(self.config.mod_key), + |wm, _| wm.handle_switch_stack(), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("Print", ShiftMask), - |wm, _| wm.spawn("screenshot.sh", &["-edit"]), - )); - - self.add_keybind(KeyBinding::new( - self.xlib.make_key("M", self.config.mod_key), - Self::handle_switch_stack, - )); - - self.add_keybind(KeyBinding::new( - self.xlib.make_key("F", self.config.mod_key), + KeyBind::new(VirtualKeyCode::F).with_mod(self.config.mod_key), |wm, _| { wm.clients .get_focused() @@ -158,45 +226,55 @@ impl WindowManager { )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("Q", self.config.mod_key), - Self::kill_client, + KeyBind::new(VirtualKeyCode::Q).with_mod(self.config.mod_key), + |wm, _| wm.kill_client(), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("Q", self.config.mod_key | ShiftMask), + KeyBind::new(VirtualKeyCode::Q) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), |wm, _| wm.quit(), )); self.add_keybind(KeyBinding::new( - self.xlib - .make_key("Return", self.config.mod_key | ShiftMask), - |wm, _| wm.spawn("alacritty", &[]), + KeyBind::new(VirtualKeyCode::Return) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), + |wm, _| { + wm.spawn( + &wm.config.terminal_command.0, + &wm.config.terminal_command.1, + ) + }, )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("J", self.config.mod_key), + KeyBind::new(VirtualKeyCode::J).with_mod(self.config.mod_key), |wm, _| wm.move_focus(Direction::south()), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("K", self.config.mod_key), + KeyBind::new(VirtualKeyCode::K).with_mod(self.config.mod_key), |wm, _| wm.move_focus(Direction::north()), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("H", self.config.mod_key), + KeyBind::new(VirtualKeyCode::H).with_mod(self.config.mod_key), |wm, _| wm.move_focus(Direction::west()), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("L", self.config.mod_key), + KeyBind::new(VirtualKeyCode::L).with_mod(self.config.mod_key), |wm, _| wm.move_focus(Direction::east()), )); // resize master stack self.add_keybind(KeyBinding::new( - self.xlib.make_key("K", self.config.mod_key | ShiftMask), + KeyBind::new(VirtualKeyCode::K) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), |wm, _| { wm.clients.change_master_size(0.1); wm.arrange_clients(); @@ -204,7 +282,9 @@ impl WindowManager { )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("J", self.config.mod_key | ShiftMask), + KeyBind::new(VirtualKeyCode::J) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), |wm, _| { wm.clients.change_master_size(-0.1); wm.arrange_clients(); @@ -213,53 +293,57 @@ impl WindowManager { self.add_vs_switch_keybinds(); - self.xlib.init(); + self.backend.set_active_window_border_color( + &self.config.active_window_border_color, + ); + self.backend.set_inactive_window_border_color( + &self.config.inactive_window_border_color, + ); + + // add all already existing windows to the WM + if let Some(windows) = self.backend.all_windows() { + windows + .into_iter() + .for_each(|window| self.new_client(window)); + } self } - fn add_keybind(&mut self, keybind: KeyBinding) { - self.xlib.add_global_keybind(keybind.key); - self.keybinds.push(keybind); + fn add_keybind(&mut self, keybind: KeyBinding) { + self.backend.add_keybind((&keybind.key).into()); + self.keybinds.borrow_mut().push(keybind); } fn add_vs_switch_keybinds(&mut self) { - fn rotate_west(wm: &mut WindowManager, _: &XKeyEvent) { - wm.rotate_virtual_screen(Direction::West(N)); - } - - fn rotate_east(wm: &mut WindowManager, _: &XKeyEvent) { - wm.rotate_virtual_screen(Direction::East(N)); - } - - fn goto_nth(wm: &mut WindowManager, _: &XKeyEvent) { - wm.go_to_nth_virtual_screen(N) - } - // Old keybinds self.add_keybind(KeyBinding::new( - self.xlib.make_key("Left", self.config.mod_key), - rotate_west::<1>, + KeyBind::new(VirtualKeyCode::Left).with_mod(self.config.mod_key), + |wm, _| wm.rotate_virtual_screen(Direction::West(1)), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("H", self.config.mod_key | ShiftMask), - rotate_west::<1>, + KeyBind::new(VirtualKeyCode::H) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), + |wm, _| wm.rotate_virtual_screen(Direction::West(1)), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("Right", self.config.mod_key), - rotate_east::<1>, + KeyBind::new(VirtualKeyCode::Right).with_mod(self.config.mod_key), + |wm, _| wm.rotate_virtual_screen(Direction::East(1)), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("L", self.config.mod_key | ShiftMask), - rotate_east::<1>, + KeyBind::new(VirtualKeyCode::L) + .with_mod(self.config.mod_key) + .with_mod(ModifierKey::Shift), + |wm, _| wm.rotate_virtual_screen(Direction::East(1)), )); self.add_keybind(KeyBinding::new( - self.xlib.make_key("Tab", self.config.mod_key), + KeyBind::new(VirtualKeyCode::Tab).with_mod(self.config.mod_key), |wm, _| wm.rotate_virtual_screen_back(), )); @@ -267,113 +351,153 @@ impl WindowManager { // Press Mod + `1` to move go to the `1`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("1", self.config.mod_key), - goto_nth::<1>, + KeyBind::new(VirtualKeyCode::One).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(1), )); // Press Mod + `2` to move go to the `2`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("2", self.config.mod_key), - goto_nth::<2>, + KeyBind::new(VirtualKeyCode::Two).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(2), )); // Press Mod + `3` to move go to the `3`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("3", self.config.mod_key), - goto_nth::<3>, + KeyBind::new(VirtualKeyCode::Three).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(3), )); // Press Mod + `4` to move go to the `4`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("4", self.config.mod_key), - goto_nth::<4>, + KeyBind::new(VirtualKeyCode::Four).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(4), )); // Press Mod + `5` to move go to the `5`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("5", self.config.mod_key), - goto_nth::<5>, + KeyBind::new(VirtualKeyCode::Five).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(5), )); // Press Mod + `6` to move go to the `6`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("6", self.config.mod_key), - goto_nth::<6>, + KeyBind::new(VirtualKeyCode::Six).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(6), )); // Press Mod + `7` to move go to the `7`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("7", self.config.mod_key), - goto_nth::<7>, + KeyBind::new(VirtualKeyCode::Seven).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(7), )); // Press Mod + `8` to move go to the `8`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("8", self.config.mod_key), - goto_nth::<8>, + KeyBind::new(VirtualKeyCode::Eight).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(8), )); // Press Mod + `9` to move go to the `9`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("9", self.config.mod_key), - goto_nth::<9>, + KeyBind::new(VirtualKeyCode::Nine).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(9), )); // Press Mod + `0` to move go to the `0`th virtual screen self.add_keybind(KeyBinding::new( - self.xlib.make_key("0", self.config.mod_key), - goto_nth::<10>, + KeyBind::new(VirtualKeyCode::Zero).with_mod(self.config.mod_key), + |wm, _| wm.go_to_nth_virtual_screen(10), )); } + #[allow(unused_mut)] pub fn run(mut self) -> ! { loop { - let event = self.xlib.next_event(); + let event = self.backend.next_event(); - match event.get_type() { - xlib::MapRequest => self.map_request(&event), - xlib::UnmapNotify => self.unmap_notify(&event), - xlib::ConfigureRequest => self.configure_request(&event), - xlib::EnterNotify => self.enter_notify(&event), - xlib::DestroyNotify => self.destroy_notify(&event), - xlib::ButtonPress => self.button_press(event.as_ref()), - xlib::ButtonRelease => self.button_release(event.as_ref()), - xlib::MotionNotify => self.motion_notify(event.as_ref()), - xlib::KeyPress => self.handle_keybinds(event.as_ref()), + match event { + WindowEvent::KeyEvent(event) => { + if event.state == KeyState::Pressed { + self.handle_keybinds(&event); + } + } + WindowEvent::ButtonEvent(event) => { + self.button_event(&event); + } + WindowEvent::MapRequestEvent(MapEvent { window }) => { + if !self.clients.contains(&window) { + self.new_client(window); + } + + self.backend.handle_event(event); + } + WindowEvent::UnmapEvent(event) => { + self.clients.remove(&event.window); + self.arrange_clients(); + } + WindowEvent::EnterEvent(event) => { + self.focus_client(&event.window, false); + } + WindowEvent::MotionEvent(event) => { + self.do_move_resize_window(&event); + } + WindowEvent::ConfigureEvent(ConfigureEvent { + window, .. + }) => { + if !self.clients.contains(&window) { + self.backend.handle_event(event); + } + // TODO + // match self.clients.get(&event.window).into_option() { + // Some(client) => self + // .xlib + // .configure_client(client, self.clients.get_border()), + // None => self.xlib.configure_window(event), + // } + } + + // i dont think i actually have to handle destroy notify events. + // every window should be unmapped regardless + // xlib::DestroyNotify => self.destroy_notify(&event), _ => {} } } } fn quit(&self) -> ! { - self.xlib.close_dpy(); + // TODO: should the window manager kill all clients on exit? probably + if self.config.kill_clients_on_exit { + self.clients + .iter_all_clients() + .for_each(|(&window, _)| self.backend.kill_window(window)); + } info!("Goodbye."); std::process::exit(0); } - fn kill_client(&mut self, _event: &XKeyEvent) { + fn kill_client(&mut self) { if let Some(client) = self.clients.get_focused().into_option() { - self.xlib.kill_client(client); + self.backend.kill_window(client.window); } } // TODO: change this somehow cuz I'm not a big fan of this "hardcoded" keybind stuff - fn handle_keybinds(&mut self, event: &XKeyEvent) { - let clean_mask = self.xlib.get_clean_mask(); - for kb in self.keybinds.clone().into_iter() { - if let KeyOrButton::Key(keycode, modmask) = kb.key { - if keycode as u32 == event.keycode - && modmask & clean_mask == event.state & clean_mask - { - (kb.closure)(self, event); - } + fn handle_keybinds(&mut self, event: &KeyEvent) { + // I'm not sure if this has to be a Rc> or if it would be better as a Cell<> + let keybinds = self.keybinds.clone(); + + for kb in keybinds.borrow().iter() { + if kb.key.key == event.keycode + && kb.key.modifiers == event.modifierstate + { + kb.call(self, event); } } } - fn handle_switch_stack(&mut self, _event: &XKeyEvent) { + fn handle_switch_stack(&mut self) { if let Some(client) = self.clients.get_focused().into_option().map(|c| c.key()) { @@ -508,22 +632,23 @@ impl WindowManager { fn hide_hidden_clients(&self) { self.clients .iter_hidden() - .for_each(|(_, c)| self.xlib.hide_client(c)); + .for_each(|(_, c)| self.backend.hide_window(c.window)); } fn raise_floating_clients(&self) { self.clients .iter_floating() - .for_each(|(_, c)| self.xlib.raise_client(c)); + .for_each(|(_, c)| self.backend.raise_window(c.window)); self.clients .iter_transient() - .for_each(|(_, c)| self.xlib.raise_client(c)); + .for_each(|(_, c)| self.backend.raise_window(c.window)); } fn arrange_clients(&mut self) { self.clients.iter_visible().for_each(|(_, c)| { - self.xlib.move_resize_client(c); + self.backend.move_window(c.window, c.position); + self.backend.resize_window(c.window, c.size); //self.xlib.expose_client(c); }); @@ -548,19 +673,19 @@ impl WindowManager { let (new, old) = self.clients.focus_client(key); if let Some(old) = old.into_option() { - self.xlib.unfocus_client(old); + self.backend.unfocus_window(old.window); } match new { ClientEntry::Floating(new) => { - self.xlib.focus_client(new); + self.backend.focus_window(new.window); if try_raise { - self.xlib.raise_client(new); + self.backend.raise_window(new.window); } } ClientEntry::Tiled(new) => { - self.xlib.focus_client(new); + self.backend.focus_window(new.window); } _ => {} } @@ -569,81 +694,44 @@ impl WindowManager { fn new_client(&mut self, window: Window) { info!("new client: {:?}", window); let client = if let Some(transient_window) = - self.xlib.get_transient_for_window(window) + self.backend.get_parent_window(window) { Client::new_transient( window, - self.xlib.get_window_size(window).unwrap_or((100, 100)), + self.backend + .get_window_size(window) + .unwrap_or((100, 100).into()), transient_window, ) } else { Client::new_default(window) }; - self.xlib - .configure_client(&client, self.clients.get_border()); + self.backend.configure_window( + window, + None, + None, + Some(self.clients.get_border()), + ); self.clients.insert(client).unwrap(); self.arrange_clients(); - self.xlib.map_window(window); - self.focus_client(&window, true); } - fn map_request(&mut self, event: &XEvent) { - let event: &XMapRequestEvent = event.as_ref(); - - if !self.clients.contains(&event.window) { - self.new_client(event.window); - } - } - - fn unmap_notify(&mut self, event: &XEvent) { - let event: &XUnmapEvent = event.as_ref(); - - self.clients.remove(&event.window); - - self.arrange_clients(); - } - - fn destroy_notify(&mut self, event: &XEvent) { - let event: &XDestroyWindowEvent = event.as_ref(); - - self.clients.remove(&event.window); - - self.arrange_clients(); - } - - fn configure_request(&mut self, event: &XEvent) { - let event: &XConfigureRequestEvent = event.as_ref(); - - match self.clients.get(&event.window).into_option() { - Some(client) => self - .xlib - .configure_client(client, self.clients.get_border()), - None => self.xlib.configure_window(event), - } - } - - fn enter_notify(&mut self, event: &XEvent) { - let event: &XCrossingEvent = event.as_ref(); - - self.focus_client(&event.window, false); - } - /// ensure event.subwindow refers to a valid client. - fn start_move_resize_window(&mut self, event: &XButtonPressedEvent) { - let window = event.subwindow; + fn start_move_resize_window(&mut self, event: &ButtonEvent) { + let window = event.window; // xev.subwindow - match event.button { - 1 => { + match event.keycode { + MouseButton::Left => { if self.clients.set_floating(&window) { self.arrange_clients(); } self.move_resize_window = MoveResizeInfo::Move(MoveInfoInner { window, - starting_cursor_pos: (event.x, event.y), + starting_cursor_pos: event.cursor_position, starting_window_pos: self .clients .get(&window) @@ -651,7 +739,7 @@ impl WindowManager { .position, }); } - 3 => { + MouseButton::Right => { if self.clients.set_floating(&window) { self.arrange_clients(); } @@ -660,18 +748,18 @@ impl WindowManager { let corner_pos = { ( - client.position.0 + client.size.0, - client.position.1 + client.size.1, + client.position.x + client.size.x, + client.position.y + client.size.y, ) }; - self.xlib.move_cursor(None, corner_pos); - self.xlib.grab_cursor(); + self.backend.move_cursor(None, corner_pos.into()); + self.backend.grab_cursor(); self.move_resize_window = MoveResizeInfo::Resize(ResizeInfoInner { window, - starting_cursor_pos: corner_pos, + starting_cursor_pos: corner_pos.into(), starting_window_size: client.size, }); } @@ -679,21 +767,25 @@ impl WindowManager { } } - fn end_move_resize_window(&mut self, event: &XButtonReleasedEvent) { - if event.button == 1 || event.button == 3 { - self.move_resize_window = MoveResizeInfo::None; - } - if event.button == 3 { - self.xlib.release_cursor(); + fn end_move_resize_window(&mut self, event: &ButtonEvent) { + match event.keycode { + MouseButton::Left => { + self.move_resize_window = MoveResizeInfo::None; + } + MouseButton::Right => { + self.move_resize_window = MoveResizeInfo::None; + self.backend.ungrab_cursor(); + } + _ => {} } } - fn do_move_resize_window(&mut self, event: &XMotionEvent) { + fn do_move_resize_window(&mut self, event: &MotionEvent) { match &self.move_resize_window { MoveResizeInfo::Move(info) => { let (x, y) = ( - event.x - info.starting_cursor_pos.0, - event.y - info.starting_cursor_pos.1, + event.position.x - info.starting_cursor_pos.x, + event.position.y - info.starting_cursor_pos.y, ); if let Some(client) = @@ -701,16 +793,18 @@ impl WindowManager { { let position = &mut client.position; - position.0 = info.starting_window_pos.0 + x; - position.1 = info.starting_window_pos.1 + y; + position.x = info.starting_window_pos.x + x; + position.y = info.starting_window_pos.y + y; - self.xlib.move_client(client); + self.backend.move_window(client.window, client.position); } } MoveResizeInfo::Resize(info) => { + info!("do_resize: {:#?}", info); + let (x, y) = ( - event.x - info.starting_cursor_pos.0, - event.y - info.starting_cursor_pos.1, + event.position.x - info.starting_cursor_pos.x, + event.position.y - info.starting_cursor_pos.y, ); if let Some(client) = @@ -718,85 +812,76 @@ impl WindowManager { { let size = &mut client.size; - size.0 = std::cmp::max(1, info.starting_window_size.0 + x); - size.1 = std::cmp::max(1, info.starting_window_size.1 + y); + size.x = std::cmp::max(1, info.starting_window_size.x + x); + size.y = std::cmp::max(1, info.starting_window_size.y + y); - self.xlib.resize_client(client); + self.backend.resize_window(client.window, client.size); } } _ => {} } } - fn button_press(&mut self, event: &XButtonPressedEvent) { - self.focus_client(&event.subwindow, true); + fn button_event(&mut self, event: &ButtonEvent) { + match event.state { + KeyState::Pressed => { + self.focus_client(&event.window, true); - match event.button { - 1 | 3 => match self.move_resize_window { - MoveResizeInfo::None - if self - .xlib - .are_masks_equal(event.state, self.config.mod_key) - && self.clients.contains(&event.subwindow) => - { - self.start_move_resize_window(event) + match event.keycode { + MouseButton::Left | MouseButton::Right => { + match self.move_resize_window { + MoveResizeInfo::None + if ModifierState::from([self + .config + .mod_key]) + .eq(&event.modifierstate) + && self.clients.contains(&event.window) => + { + self.start_move_resize_window(event) + } + _ => {} + } + } + MouseButton::Middle => { + self.clients.toggle_floating(&event.window); + self.arrange_clients(); + } + _ => {} + } + } + KeyState::Released => match self.move_resize_window { + MoveResizeInfo::None => {} + _ => { + self.end_move_resize_window(event); } - _ => {} }, - 2 => { - self.clients.toggle_floating(&event.subwindow); - self.arrange_clients(); - } - _ => {} } } - fn button_release(&mut self, event: &XButtonReleasedEvent) { - match self.move_resize_window { - MoveResizeInfo::None => {} - _ => { - self.end_move_resize_window(event); - } - } - } - - fn motion_notify(&mut self, event: &XMotionEvent) { - self.do_move_resize_window(event); - } - - pub fn spawn(&self, command: &str, args: &[&str]) { - info!("spawn: {:?} {:?}", command, args.join(" ")); - match std::process::Command::new(command).args(args).spawn() { + pub fn spawn<'a, S, I>(&self, command: S, args: I) + where + S: AsRef + AsRef, + I: IntoIterator + std::fmt::Debug, + { + info!("spawn: {:?} {:?}", AsRef::::as_ref(&command), args); + match std::process::Command::new(AsRef::::as_ref( + &command, + )) + .args(args) + .spawn() + { Ok(_) => {} Err(err) => { - error!("Failed to spawn {:?}: {:?}", command, err); + error!( + "Failed to spawn {:?}: {:?}", + AsRef::::as_ref(&command), + err + ); } } } } -impl KeyBinding { - fn new(key: KeyOrButton, closure: F) -> Self - where - F: Fn(&mut WindowManager, &XKeyEvent) + 'static, - { - Self { - key, - closure: Rc::new(closure), - } - } -} - -impl Default for WMConfig { - fn default() -> Self { - Self { - num_virtualscreens: 10, - mod_key: Mod4Mask, - gap: Some(2), - } - } -} - impl Direction { fn west() -> Self { Direction::West(1) diff --git a/src/xlib.rs b/src/xlib.rs index a5c8f6f..4c3b247 100644 --- a/src/xlib.rs +++ b/src/xlib.rs @@ -53,6 +53,12 @@ impl KeyOrButton { #[derive(Clone)] pub struct Display(Rc<*mut xlib::Display>); +impl Drop for XLib { + fn drop(&mut self) { + self.close_dpy(); + } +} + impl XLib { pub fn new() -> Self { let (display, _screen, root) = unsafe { @@ -268,16 +274,16 @@ impl XLib { pub fn move_resize_client(&self, client: &Client) { let mut windowchanges = xlib::XWindowChanges { - x: client.position.0, - y: client.position.1, - width: client.size.0, - height: client.size.1, + x: client.position.x, + y: client.position.y, + width: client.size.x, + height: client.size.y, border_width: 0, sibling: 0, stack_mode: 0, }; - if client.size.0 < 1 || client.size.1 < 1 { + if client.size.x < 1 || client.size.y < 1 { error!("client {:?} size is less than 1 pixel!", client); } else { unsafe { @@ -297,16 +303,16 @@ impl XLib { pub fn move_client(&self, client: &Client) { let mut wc = xlib::XWindowChanges { - x: client.position.0, - y: client.position.1, - width: client.size.0, - height: client.size.1, + x: client.position.x, + y: client.position.y, + width: client.size.x, + height: client.size.y, border_width: 0, sibling: 0, stack_mode: 0, }; - if client.size.0 < 1 || client.size.1 < 1 { + if client.size.x < 1 || client.size.y < 1 { error!("client {:?} size is less than 1 pixel!", client); } else { unsafe { @@ -322,16 +328,16 @@ impl XLib { pub fn resize_client(&self, client: &Client) { let mut wc = xlib::XWindowChanges { - x: client.position.0, - y: client.position.1, - width: client.size.0, - height: client.size.1, + x: client.position.x, + y: client.position.y, + width: client.size.x, + height: client.size.y, border_width: 0, sibling: 0, stack_mode: 0, }; - if client.size.0 < 1 || client.size.1 < 1 { + if client.size.x < 1 || client.size.y < 1 { error!("client {:?} size is less than 1 pixel!", client); } else { unsafe { @@ -347,16 +353,16 @@ impl XLib { pub fn hide_client(&self, client: &Client) { let mut wc = xlib::XWindowChanges { - x: client.size.0 * -2, - y: client.position.1, - width: client.size.0, - height: client.size.1, + x: client.size.x * -2, + y: client.position.y, + width: client.size.x, + height: client.size.y, border_width: 0, sibling: 0, stack_mode: 0, }; - if client.size.0 < 1 || client.size.1 < 1 { + if client.size.x < 1 || client.size.y < 1 { error!("client {:?} size is less than 1 pixel!", client); } else { unsafe { @@ -391,6 +397,7 @@ impl XLib { } } + #[allow(dead_code)] fn get_window_attributes( &self, window: Window, @@ -421,10 +428,12 @@ impl XLib { } } + #[allow(dead_code)] pub fn expose_client(&self, client: &Client) { self.expose_window(client.window); } + #[allow(dead_code)] fn expose_window(&self, window: Window) { if let Some(wa) = self.get_window_attributes(window) { unsafe { @@ -468,10 +477,10 @@ impl XLib { display: self.dpy(), event: client.window, window: client.window, - x: client.position.0, - y: client.position.1, - width: client.size.0, - height: client.size.1, + x: client.position.x, + y: client.position.y, + width: client.size.x, + height: client.size.y, border_width: border, override_redirect: 0, send_event: 0, @@ -527,7 +536,7 @@ impl XLib { } } - pub fn close_dpy(&self) { + fn close_dpy(&self) { unsafe { XCloseDisplay(self.dpy()); } diff --git a/starship.jpg b/starship.jpg deleted file mode 100644 index 40b2bb0..0000000 Binary files a/starship.jpg and /dev/null differ diff --git a/xinitrc b/xinitrc deleted file mode 100644 index 4775a62..0000000 --- a/xinitrc +++ /dev/null @@ -1,8 +0,0 @@ -#!/bin/sh - -xset b off -xsetroot -solid darkslategrey -feh --bg-fill "/mnt/storage/rust/wm/starship.jpg" -xset r rate 250 30 -export RUST_BACKTRACE=1 -exec /mnt/storage/code/rust/wm/target/release/wm 2>&1 /home/user/.local/portlights.log