autocommit 2022-01-09 16:13:56 CET
This commit is contained in:
parent
928f94daa6
commit
b08bd94353
13 changed files with 151 additions and 28 deletions
6
build.rs
6
build.rs
|
|
@ -71,6 +71,12 @@ fn main() -> anyhow::Result<()> {
|
||||||
"xkb_keymap_compile_flags",
|
"xkb_keymap_compile_flags",
|
||||||
)?;
|
)?;
|
||||||
write_ty(&mut f, xkbcommon::XKB_KEYMAP_FORMAT, "xkb_keymap_format")?;
|
write_ty(&mut f, xkbcommon::XKB_KEYMAP_FORMAT, "xkb_keymap_format")?;
|
||||||
|
write_ty(
|
||||||
|
&mut f,
|
||||||
|
xkbcommon::XKB_STATE_COMPONENT,
|
||||||
|
"xkb_state_component",
|
||||||
|
)?;
|
||||||
|
write_ty(&mut f, xkbcommon::XKB_KEY_DIRECTION, "xkb_key_direction")?;
|
||||||
|
|
||||||
println!("cargo:rerun-if-changed=build.rs");
|
println!("cargo:rerun-if-changed=build.rs");
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
|
||||||
|
|
@ -8,6 +8,7 @@ use crate::ifs::wl_surface::WlSurface;
|
||||||
use crate::pixman::{Image, PixmanError};
|
use crate::pixman::{Image, PixmanError};
|
||||||
use crate::servermem::{ServerMem, ServerMemError};
|
use crate::servermem::{ServerMem, ServerMemError};
|
||||||
use crate::tree::{Node, NodeKind, ToplevelNode};
|
use crate::tree::{Node, NodeKind, ToplevelNode};
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::utils::copyhashmap::CopyHashMap;
|
use crate::utils::copyhashmap::CopyHashMap;
|
||||||
use crate::utils::ptr_ext::PtrExt;
|
use crate::utils::ptr_ext::PtrExt;
|
||||||
use crate::wheel::{WheelDispatcher, WheelId};
|
use crate::wheel::{WheelDispatcher, WheelId};
|
||||||
|
|
@ -24,7 +25,6 @@ use uapi::c;
|
||||||
use xcb_dl::{ffi, Xcb, XcbShm, XcbXinput, XcbXkb};
|
use xcb_dl::{ffi, Xcb, XcbShm, XcbXinput, XcbXkb};
|
||||||
use xcb_dl_util::error::{XcbError, XcbErrorParser};
|
use xcb_dl_util::error::{XcbError, XcbErrorParser};
|
||||||
use xcb_dl_util::xcb_box::XcbBox;
|
use xcb_dl_util::xcb_box::XcbBox;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
#[derive(Debug, Error)]
|
#[derive(Debug, Error)]
|
||||||
pub enum XorgBackendError {
|
pub enum XorgBackendError {
|
||||||
|
|
|
||||||
|
|
@ -8,12 +8,12 @@ use crate::ifs::wl_surface::xdg_surface::{XdgSurface, XdgSurfaceId};
|
||||||
use crate::ifs::wl_surface::{WlSurface, WlSurfaceId};
|
use crate::ifs::wl_surface::{WlSurface, WlSurfaceId};
|
||||||
use crate::ifs::xdg_wm_base::{XdgWmBaseId, XdgWmBaseObj};
|
use crate::ifs::xdg_wm_base::{XdgWmBaseId, XdgWmBaseObj};
|
||||||
use crate::object::{Object, ObjectId};
|
use crate::object::{Object, ObjectId};
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::utils::copyhashmap::CopyHashMap;
|
use crate::utils::copyhashmap::CopyHashMap;
|
||||||
use ahash::AHashMap;
|
use ahash::AHashMap;
|
||||||
use std::cell::{RefCell, RefMut};
|
use std::cell::{RefCell, RefMut};
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
pub struct Objects {
|
pub struct Objects {
|
||||||
pub display: CloneCell<Option<Rc<WlDisplay>>>,
|
pub display: CloneCell<Option<Rc<WlDisplay>>>,
|
||||||
|
|
|
||||||
|
|
@ -13,8 +13,9 @@ use crate::ifs::wl_seat::wl_touch::WlTouch;
|
||||||
use crate::object::{Interface, Object, ObjectId};
|
use crate::object::{Interface, Object, ObjectId};
|
||||||
use crate::tree::{Node, NodeBase, NodeKind, ToplevelNode};
|
use crate::tree::{Node, NodeBase, NodeKind, ToplevelNode};
|
||||||
use crate::utils::buffd::MsgParser;
|
use crate::utils::buffd::MsgParser;
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::utils::copyhashmap::CopyHashMap;
|
use crate::utils::copyhashmap::CopyHashMap;
|
||||||
use crate::xkbcommon::XkbContext;
|
use crate::xkbcommon::{ModifierState, XkbContext, XkbState, XKB_KEY_DOWN, XKB_KEY_UP};
|
||||||
use crate::State;
|
use crate::State;
|
||||||
use ahash::{AHashMap, AHashSet};
|
use ahash::{AHashMap, AHashSet};
|
||||||
use bstr::ByteSlice;
|
use bstr::ByteSlice;
|
||||||
|
|
@ -23,7 +24,6 @@ use std::io::Write;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
pub use types::*;
|
pub use types::*;
|
||||||
use uapi::{c, OwnedFd};
|
use uapi::{c, OwnedFd};
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
id!(WlSeatId);
|
id!(WlSeatId);
|
||||||
|
|
||||||
|
|
@ -57,15 +57,17 @@ pub struct WlSeatGlobal {
|
||||||
keyboard_node: CloneCell<Rc<dyn Node>>,
|
keyboard_node: CloneCell<Rc<dyn Node>>,
|
||||||
pressed_keys: RefCell<AHashSet<u32>>,
|
pressed_keys: RefCell<AHashSet<u32>>,
|
||||||
bindings: RefCell<AHashMap<ClientId, AHashMap<WlSeatId, Rc<WlSeatObj>>>>,
|
bindings: RefCell<AHashMap<ClientId, AHashMap<WlSeatId, Rc<WlSeatObj>>>>,
|
||||||
|
kb_state: RefCell<XkbState>,
|
||||||
layout: Rc<OwnedFd>,
|
layout: Rc<OwnedFd>,
|
||||||
layout_size: u32,
|
layout_size: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WlSeatGlobal {
|
impl WlSeatGlobal {
|
||||||
pub fn new(name: GlobalName, state: &Rc<State>, seat: &Rc<dyn Seat>) -> Self {
|
pub fn new(name: GlobalName, state: &Rc<State>, seat: &Rc<dyn Seat>) -> Self {
|
||||||
let (layout, layout_size) = {
|
let (kb_state, layout, layout_size) = {
|
||||||
let ctx = XkbContext::new().unwrap();
|
let ctx = XkbContext::new().unwrap();
|
||||||
let keymap = ctx.default_keymap().unwrap();
|
let keymap = ctx.default_keymap().unwrap();
|
||||||
|
let state = keymap.state().unwrap();
|
||||||
let string = keymap.as_str().unwrap();
|
let string = keymap.as_str().unwrap();
|
||||||
let mut memfd =
|
let mut memfd =
|
||||||
uapi::memfd_create("keymap", c::MFD_CLOEXEC | c::MFD_ALLOW_SEALING).unwrap();
|
uapi::memfd_create("keymap", c::MFD_CLOEXEC | c::MFD_ALLOW_SEALING).unwrap();
|
||||||
|
|
@ -77,7 +79,7 @@ impl WlSeatGlobal {
|
||||||
c::F_SEAL_SEAL | c::F_SEAL_GROW | c::F_SEAL_SHRINK | c::F_SEAL_WRITE,
|
c::F_SEAL_SEAL | c::F_SEAL_GROW | c::F_SEAL_SHRINK | c::F_SEAL_WRITE,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
(Rc::new(memfd), (string.len() + 1) as _)
|
(state, Rc::new(memfd), (string.len() + 1) as _)
|
||||||
};
|
};
|
||||||
Self {
|
Self {
|
||||||
name,
|
name,
|
||||||
|
|
@ -91,6 +93,7 @@ impl WlSeatGlobal {
|
||||||
keyboard_node: CloneCell::new(state.root.clone()),
|
keyboard_node: CloneCell::new(state.root.clone()),
|
||||||
pressed_keys: RefCell::new(Default::default()),
|
pressed_keys: RefCell::new(Default::default()),
|
||||||
bindings: Default::default(),
|
bindings: Default::default(),
|
||||||
|
kb_state: RefCell::new(kb_state),
|
||||||
layout,
|
layout,
|
||||||
layout_size,
|
layout_size,
|
||||||
}
|
}
|
||||||
|
|
@ -244,6 +247,16 @@ impl WlSeatGlobal {
|
||||||
if let NodeKind::Toplevel(tl) = kb_node.clone().into_kind() {
|
if let NodeKind::Toplevel(tl) = kb_node.clone().into_kind() {
|
||||||
self.tl_kb_event(&tl, |k| k.leave(0, tl.surface.surface.surface.id))
|
self.tl_kb_event(&tl, |k| k.leave(0, tl.surface.surface.surface.id))
|
||||||
.await;
|
.await;
|
||||||
|
let ModifierState {
|
||||||
|
mods_depressed,
|
||||||
|
mods_latched,
|
||||||
|
mods_locked,
|
||||||
|
group,
|
||||||
|
} = self.kb_state.borrow().mods();
|
||||||
|
self.tl_kb_event(&tl, |k| {
|
||||||
|
k.modifiers(0, mods_depressed, mods_latched, mods_locked, group)
|
||||||
|
})
|
||||||
|
.await;
|
||||||
}
|
}
|
||||||
self.keyboard_node.set(node.clone());
|
self.keyboard_node.set(node.clone());
|
||||||
}
|
}
|
||||||
|
|
@ -281,28 +294,39 @@ impl WlSeatGlobal {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn key_event(&self, key: u32, state: KeyState) {
|
async fn key_event(&self, key: u32, state: KeyState) {
|
||||||
let state = {
|
let (state, xkb_dir) = {
|
||||||
let mut pk = self.pressed_keys.borrow_mut();
|
let mut pk = self.pressed_keys.borrow_mut();
|
||||||
match state {
|
match state {
|
||||||
KeyState::Released => {
|
KeyState::Released => {
|
||||||
if !pk.remove(&key) {
|
if !pk.remove(&key) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
log::info!("release");
|
(wl_keyboard::RELEASED, XKB_KEY_UP)
|
||||||
wl_keyboard::RELEASED
|
|
||||||
}
|
}
|
||||||
KeyState::Pressed => {
|
KeyState::Pressed => {
|
||||||
if !pk.insert(key) {
|
if !pk.insert(key) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
log::info!("press");
|
(wl_keyboard::PRESSED, XKB_KEY_DOWN)
|
||||||
wl_keyboard::PRESSED
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
let mods = self.kb_state.borrow_mut().update(key, xkb_dir);
|
||||||
let node = self.keyboard_node.get().into_kind();
|
let node = self.keyboard_node.get().into_kind();
|
||||||
if let NodeKind::Toplevel(node) = node {
|
if let NodeKind::Toplevel(node) = node {
|
||||||
self.tl_kb_event(&node, |k| k.key(0, 0, key, state)).await;
|
self.tl_kb_event(&node, |k| k.key(0, 0, key, state)).await;
|
||||||
|
if let Some(mods) = mods {
|
||||||
|
self.tl_kb_event(&node, |k| {
|
||||||
|
k.modifiers(
|
||||||
|
0,
|
||||||
|
mods.mods_depressed,
|
||||||
|
mods.mods_latched,
|
||||||
|
mods.mods_locked,
|
||||||
|
mods.group,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -5,10 +5,10 @@ use crate::clientmem::ClientMem;
|
||||||
use crate::ifs::wl_buffer::WlBuffer;
|
use crate::ifs::wl_buffer::WlBuffer;
|
||||||
use crate::object::{Interface, Object, ObjectId};
|
use crate::object::{Interface, Object, ObjectId};
|
||||||
use crate::utils::buffd::MsgParser;
|
use crate::utils::buffd::MsgParser;
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
pub use types::*;
|
pub use types::*;
|
||||||
use uapi::OwnedFd;
|
use uapi::OwnedFd;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
const CREATE_BUFFER: u32 = 0;
|
const CREATE_BUFFER: u32 = 0;
|
||||||
const DESTROY: u32 = 1;
|
const DESTROY: u32 = 1;
|
||||||
|
|
@ -75,7 +75,8 @@ impl WlShmPool {
|
||||||
if (req.size as usize) < self.mem.get().len() {
|
if (req.size as usize) < self.mem.get().len() {
|
||||||
return Err(ResizeError::CannotShrink);
|
return Err(ResizeError::CannotShrink);
|
||||||
}
|
}
|
||||||
self.mem.set(Rc::new(ClientMem::new(self.fd.raw(), req.size as usize)?));
|
self.mem
|
||||||
|
.set(Rc::new(ClientMem::new(self.fd.raw(), req.size as usize)?));
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,7 @@ use crate::object::{Interface, Object, ObjectId};
|
||||||
use crate::pixman::Region;
|
use crate::pixman::Region;
|
||||||
use crate::tree::{NodeBase, NodeCommon, ToplevelNode};
|
use crate::tree::{NodeBase, NodeCommon, ToplevelNode};
|
||||||
use crate::utils::buffd::{MsgParser, MsgParserError};
|
use crate::utils::buffd::{MsgParser, MsgParserError};
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::utils::copyhashmap::CopyHashMap;
|
use crate::utils::copyhashmap::CopyHashMap;
|
||||||
use crate::utils::linkedlist::{LinkedList, Node as LinkNode};
|
use crate::utils::linkedlist::{LinkedList, Node as LinkNode};
|
||||||
use ahash::AHashMap;
|
use ahash::AHashMap;
|
||||||
|
|
@ -21,7 +22,6 @@ use std::mem;
|
||||||
use std::ops::{Deref, DerefMut};
|
use std::ops::{Deref, DerefMut};
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
pub use types::*;
|
pub use types::*;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
const DESTROY: u32 = 0;
|
const DESTROY: u32 = 0;
|
||||||
const ATTACH: u32 = 1;
|
const ATTACH: u32 = 1;
|
||||||
|
|
|
||||||
|
|
@ -159,5 +159,15 @@ macro_rules! bitor {
|
||||||
Self(self.0 | rhs.0)
|
Self(self.0 | rhs.0)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl $name {
|
||||||
|
pub fn contains(self, rhs: Self) -> bool {
|
||||||
|
self.0 & rhs.0 == rhs.0
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_some(self) -> bool {
|
||||||
|
self.0 != 0
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -3,10 +3,10 @@ use crate::ifs::wl_output::WlOutputGlobal;
|
||||||
use crate::ifs::wl_seat::WlSeatGlobal;
|
use crate::ifs::wl_seat::WlSeatGlobal;
|
||||||
use crate::tree::{NodeCommon, NodeExtents, OutputNode};
|
use crate::tree::{NodeCommon, NodeExtents, OutputNode};
|
||||||
use crate::utils::asyncevent::AsyncEvent;
|
use crate::utils::asyncevent::AsyncEvent;
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::State;
|
use crate::State;
|
||||||
use std::cell::{Cell, RefCell};
|
use std::cell::{Cell, RefCell};
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
pub async fn handle_backend_events(state: Rc<State>) {
|
pub async fn handle_backend_events(state: Rc<State>) {
|
||||||
let mut beh = BackendEventHandler { state };
|
let mut beh = BackendEventHandler { state };
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,12 @@
|
||||||
use crate::backend::{Output, OutputId};
|
use crate::backend::{Output, OutputId};
|
||||||
use crate::ifs::wl_surface::xdg_surface::xdg_toplevel::XdgToplevel;
|
use crate::ifs::wl_surface::xdg_surface::xdg_toplevel::XdgToplevel;
|
||||||
|
use crate::utils::clonecell::CloneCell;
|
||||||
use crate::utils::copyhashmap::CopyHashMap;
|
use crate::utils::copyhashmap::CopyHashMap;
|
||||||
use crate::utils::linkedlist::{LinkedList, Node as LinkedNode};
|
use crate::utils::linkedlist::{LinkedList, Node as LinkedNode};
|
||||||
use ahash::AHashMap;
|
use ahash::AHashMap;
|
||||||
use std::cell::{Cell, RefCell};
|
use std::cell::{Cell, RefCell};
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use crate::utils::clonecell::CloneCell;
|
|
||||||
|
|
||||||
linear_ids!(NodeIds, NodeId);
|
linear_ids!(NodeIds, NodeId);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
|
use crate::utils::ptr_ext::{MutPtrExt, PtrExt};
|
||||||
use std::cell::UnsafeCell;
|
use std::cell::UnsafeCell;
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use crate::utils::ptr_ext::{MutPtrExt, PtrExt};
|
|
||||||
|
|
||||||
pub struct CloneCell<T: UnsafeCellCloneSafe> {
|
pub struct CloneCell<T: UnsafeCellCloneSafe> {
|
||||||
data: UnsafeCell<T>,
|
data: UnsafeCell<T>,
|
||||||
|
|
@ -16,9 +16,7 @@ impl<T: UnsafeCellCloneSafe> CloneCell<T> {
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub fn get(&self) -> T {
|
pub fn get(&self) -> T {
|
||||||
unsafe {
|
unsafe { self.data.get().deref().clone() }
|
||||||
self.data.get().deref().clone()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
|
|
@ -29,10 +27,11 @@ impl<T: UnsafeCellCloneSafe> CloneCell<T> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub fn take(&self) -> T where T: Default {
|
pub fn take(&self) -> T
|
||||||
unsafe {
|
where
|
||||||
mem::take(self.data.get().deref_mut())
|
T: Default,
|
||||||
}
|
{
|
||||||
|
unsafe { mem::take(self.data.get().deref_mut()) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -42,8 +41,8 @@ impl<T: Default + UnsafeCellCloneSafe> Default for CloneCell<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub unsafe trait UnsafeCellCloneSafe: Clone { }
|
pub unsafe trait UnsafeCellCloneSafe: Clone {}
|
||||||
|
|
||||||
unsafe impl<T: UnsafeCellCloneSafe> UnsafeCellCloneSafe for Option<T> { }
|
unsafe impl<T: UnsafeCellCloneSafe> UnsafeCellCloneSafe for Option<T> {}
|
||||||
|
|
||||||
unsafe impl<T: ?Sized> UnsafeCellCloneSafe for Rc<T> { }
|
unsafe impl<T: ?Sized> UnsafeCellCloneSafe for Rc<T> {}
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,6 @@
|
||||||
pub mod asyncevent;
|
pub mod asyncevent;
|
||||||
pub mod buffd;
|
pub mod buffd;
|
||||||
|
pub mod clonecell;
|
||||||
pub mod copyhashmap;
|
pub mod copyhashmap;
|
||||||
pub mod errorfmt;
|
pub mod errorfmt;
|
||||||
pub mod linkedlist;
|
pub mod linkedlist;
|
||||||
|
|
@ -8,4 +9,3 @@ pub mod oneshot;
|
||||||
pub mod ptr_ext;
|
pub mod ptr_ext;
|
||||||
pub mod queue;
|
pub mod queue;
|
||||||
pub mod vec_ext;
|
pub mod vec_ext;
|
||||||
pub mod clonecell;
|
|
||||||
|
|
|
||||||
|
|
@ -33,3 +33,26 @@ cenum! {
|
||||||
|
|
||||||
XKB_KEYMAP_FORMAT_TEXT_V1 = 1,
|
XKB_KEYMAP_FORMAT_TEXT_V1 = 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cenum! {
|
||||||
|
XkbStateComponent, XKB_STATE_COMPONENT;
|
||||||
|
|
||||||
|
XKB_STATE_MODS_DEPRESSED = 1 << 0,
|
||||||
|
XKB_STATE_MODS_LATCHED = 1 << 1,
|
||||||
|
XKB_STATE_MODS_LOCKED = 1 << 2,
|
||||||
|
XKB_STATE_MODS_EFFECTIVE = 1 << 3,
|
||||||
|
XKB_STATE_LAYOUT_DEPRESSED = 1 << 4,
|
||||||
|
XKB_STATE_LAYOUT_LATCHED = 1 << 5,
|
||||||
|
XKB_STATE_LAYOUT_LOCKED = 1 << 6,
|
||||||
|
XKB_STATE_LAYOUT_EFFECTIVE = 1 << 7,
|
||||||
|
XKB_STATE_LEDS = 1 << 8,
|
||||||
|
}
|
||||||
|
|
||||||
|
bitor!(XkbStateComponent);
|
||||||
|
|
||||||
|
cenum! {
|
||||||
|
XkbKeyDirection, XKB_KEY_DIRECTION;
|
||||||
|
|
||||||
|
XKB_KEY_UP = 0,
|
||||||
|
XKB_KEY_DOWN = 1,
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,8 @@ use uapi::c;
|
||||||
pub enum XkbCommonError {
|
pub enum XkbCommonError {
|
||||||
#[error("Could not create an xkbcommon context")]
|
#[error("Could not create an xkbcommon context")]
|
||||||
CreateContext,
|
CreateContext,
|
||||||
|
#[error("Could not create an xkbcommon state")]
|
||||||
|
CreateState,
|
||||||
#[error("Could not create keymap from names")]
|
#[error("Could not create keymap from names")]
|
||||||
KeymapFromNames,
|
KeymapFromNames,
|
||||||
#[error("Could not convert the keymap to a string")]
|
#[error("Could not convert the keymap to a string")]
|
||||||
|
|
@ -73,6 +75,14 @@ extern "C" {
|
||||||
) -> *mut c::c_char;
|
) -> *mut c::c_char;
|
||||||
fn xkb_keymap_unref(keymap: *mut xkb_keymap);
|
fn xkb_keymap_unref(keymap: *mut xkb_keymap);
|
||||||
fn xkb_state_unref(state: *mut xkb_state);
|
fn xkb_state_unref(state: *mut xkb_state);
|
||||||
|
fn xkb_state_new(keymap: *mut xkb_keymap) -> *mut xkb_state;
|
||||||
|
fn xkb_state_update_key(
|
||||||
|
state: *mut xkb_state,
|
||||||
|
key: u32,
|
||||||
|
direction: xkb_key_direction,
|
||||||
|
) -> xkb_state_component;
|
||||||
|
fn xkb_state_serialize_mods(state: *mut xkb_state, components: xkb_state_component) -> u32;
|
||||||
|
fn xkb_state_serialize_layout(state: *mut xkb_state, components: xkb_state_component) -> u32;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct XkbContext {
|
pub struct XkbContext {
|
||||||
|
|
@ -126,6 +136,22 @@ impl XkbKeymap {
|
||||||
s: unsafe { CStr::from_ptr(res).to_bytes().as_bstr() },
|
s: unsafe { CStr::from_ptr(res).to_bytes().as_bstr() },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn state(&self) -> Result<XkbState, XkbCommonError> {
|
||||||
|
let res = unsafe { xkb_state_new(self.keymap) };
|
||||||
|
if res.is_null() {
|
||||||
|
return Err(XkbCommonError::CreateState);
|
||||||
|
}
|
||||||
|
Ok(XkbState {
|
||||||
|
state: res,
|
||||||
|
mods: ModifierState {
|
||||||
|
mods_depressed: 0,
|
||||||
|
mods_latched: 0,
|
||||||
|
mods_locked: 0,
|
||||||
|
group: 0,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for XkbKeymap {
|
impl Drop for XkbKeymap {
|
||||||
|
|
@ -154,8 +180,42 @@ impl Drop for XkbKeymapStr {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub struct ModifierState {
|
||||||
|
pub mods_depressed: u32,
|
||||||
|
pub mods_latched: u32,
|
||||||
|
pub mods_locked: u32,
|
||||||
|
pub group: u32,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct XkbState {
|
pub struct XkbState {
|
||||||
state: *mut xkb_state,
|
state: *mut xkb_state,
|
||||||
|
mods: ModifierState,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl XkbState {
|
||||||
|
pub fn mods(&self) -> ModifierState {
|
||||||
|
self.mods
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn update(&mut self, key: u32, direction: XkbKeyDirection) -> Option<ModifierState> {
|
||||||
|
unsafe {
|
||||||
|
let changes = xkb_state_update_key(self.state, key + 8, direction.raw() as _);
|
||||||
|
if changes != 0 {
|
||||||
|
self.mods.mods_depressed =
|
||||||
|
xkb_state_serialize_mods(self.state, XKB_STATE_MODS_DEPRESSED.raw() as _);
|
||||||
|
self.mods.mods_latched =
|
||||||
|
xkb_state_serialize_mods(self.state, XKB_STATE_MODS_LATCHED.raw() as _);
|
||||||
|
self.mods.mods_locked =
|
||||||
|
xkb_state_serialize_mods(self.state, XKB_STATE_MODS_LOCKED.raw() as _);
|
||||||
|
self.mods.group =
|
||||||
|
xkb_state_serialize_layout(self.state, XKB_STATE_LAYOUT_EFFECTIVE.raw() as _);
|
||||||
|
Some(self.mods)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for XkbState {
|
impl Drop for XkbState {
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue